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SQL 基础 


SQL 简介 


SQL 是 用 于 访问 和 义理 数据 库 的 标准 的 计算 机 语言 。 


‘ta SQL? 


e SQL 指 结构 化 查询 语言 
e SQL 使 我 们 有 能 力 访问 数据 库 
。 SQL 是 一 种 ANSI 的 标准 计算 机 语言 


编者 注 : ANSI， 美 国 国家 标准 化 组 织 


SQL 能 做 什么 ? 


e。 SQL 面向 数据 库 执行 查询 

SQL 可 从 数据 库 取 回 数据 

SQL 可 在 数据 库 中 插入 新 的 记录 

SQL 可 更 新 数据 库 中 的 数据 

SQL 可 从 数据 库 删 除 记 录 

SQL 可 创建 新 数据 库 

SQL 可 在 数据 库 中 创建 新 表 

SQL 可 在 数据 库 中 创建 存储 过 程 

SQL 可 在 数据 库 中 创建 视图 

SQL 可 以 设置 表 、 存 储 过 程 和 视图 的 权限 


SQL 是 一 种 标准 - 但 是 .… 


SQL 是 一 门 ANSI 的 标准 计算 机 语言 ， 用 来 访问 和 操作 数据 库 系 统 。SQL 语句 用 于 
取 回 和 更 新 数据 库 中 的 数据 。SQL 可 与 数据 库 程 序 协同 工作 ， 比 如 MS Access, 
DB2, Informix, MS SQL Server, Oracle, Sybase 以 及 其 他 数据 库 系统 。 


不 幸 地 是 ， 存 在 着 很 多 不 同 版 本 的 SQL 语言 ， 但 是 为 了 与 ANSI 标准 相 兼 容 ， 它 
们 必须 以 相似 的 方式 共同 地 来 支持 一 些 主 要 的 关键 词 (比如 SELECT、UPDATE、 
DELETE, INSERT, WHERE 等 等 ) 。 


注释 : 除了 SQL 标准 之 外 ， 大 部 分 SQL 数据 库 程序 都 拥有 它们 自己 的 私有 扩展 ! 


在 您 的 网 站 中 使 用 SQL 


要 创建 发 布 数据 库 中 数据 的 网 站 ， 您 需要 以 下 要 素 : 
e RDBMS 数据 库 程序 (比如 MS Access, SQL Server, MySQL) 


e 服务 器 端 脚本 话 言 (比如 PHP 或 ASP) 
e SQL 
e HTML/CSS 


RDBMS 


RDBMS 指 的 是 关系 型 数据 库 管 理 系 统 。 


RDBMS 是 SQL 的 基础 ， 同 样 也 是 所 有 现代 数据 库 系 统 的 基础 ， 比 如 MS SQL 
Server, IBM DB2, Oracle, MySQL 以 及 Microsoft Access。 


RDBMS 中 的 数据 存储 在 被 称 为 表 (tables) 的 数据 库 对 象 中 。 
表 是 相关 的 数据 项 的 集合 ， 它 由 列 和 行 组 成 。 


SQL 话 法 
数据 库 表 


一 个 数据 库 通常 包含 一 个 或 多 个 表 。 每 个 表 由 一 个 名 字 标 识 ( 例 如 “客户 "或 者 " 订 
单 ") 。 表 包含 带 有 数据 的 记录 (íT) 。 


下 面 的 例子 是 一 个 名 为 "Persons" WR : 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


(每 一 条 对 应 一 个 人 ) 和 五 个 列 〈Ild、 姓 、 名 、 地 址 和 城 
市 o 


SQL 语句 


您 需要 在 数据 库 上 执行 的 大 部 分 工作 都 由 SQL 语句 完成 。 
下 面 的 语句 从 表 中 选取 LastName 列 的 数据 : 


SELECT LastName FROM Persons 


结果 集 类 似 这 样 : 
LastName 
Adams 
Bush 


Carter 


在 本 教程 中 ， 我 们 将 为 您 讲解 各 种 不 同 的 SQL 语句 。 


重要 事项 


一 定 要 记 住 ，SQL 对 大 小 写 不 敏感 ! 


SQL 话 句 后 面 的 分 号 ? 
某 些 数据 库 系 统 要 求 在 每 条 SQL 命令 的 末端 使 用 分 号 。 在 我 们 的 教程 中 不 使 用 分 
号 。 


分 号 是 在 数据 库 系 统 中 分 隔 每 条 SQL 语句 的 标准 方法 ， 这 样 就 可 以 在 对 服务 器 的 
相同 请 求 中 执行 一 条 以 上 的 语句 。 


如 果 您 使 用 的 是 MS Access 和 SQL Server 2000， 则 不 必 在 每 条 SQL 语句 之 后 使 
用 分 号 ， 不 过 某 些 数据 库 软 件 要 求 必须 使 用 分 号 。 


SQL DML 和 DDL 


可 以 把 SQL 分 为 两 个 部 分 : 数据 操作 语言 (DML) 和 数据 定义 语言 (DDL)。 


SQL (结构 化 查询 语言 ) 是 用 于 执行 查询 的 语法 。 但 是 SQL 语言 也 包含 用 于 更 新 、 
插入 和 筒 除 记录 的 语法 。 


查询 和 更 新 指令 构成 了 SQL 的 DML 部 分 : 


e SELECT - 从 数据 库 表 中 获取 数据 
e UPDATE - 更 新 数据 库 表 中 的 数据 
e DELETE - 从 数据 库 表 中 删除 数据 
e INSERT INTO - 向 数据 库 表 中 插入 数据 


SQL 的 数据 定义 语言 (DDL) 部 分 使 我 们 有 能 力 创建 或 删除 表格 。 我 们 也 可 以 定义 
索引 ( 键 ) ， 规 定 表 之 间 的 链接 ， 以 及 施加 表 间 的 约束 。 


SQL 中 最 重要 的 DDL 语句 : 


CREATE DATABASE - 创建 新 数据 库 
ALTER DATABASE - 修改 数据 库 
CREATE TABLE - 创建 新 表 

ALTER TABLE - 变更 (改变) 数据库 表 
DROP TABLE - MRK 

CREATE INDEX - 创建 索引 (搜索 键 ) 
DROP INDEX - 删除 索引 


SQL SELECT 语句 


本 章 讲解 SELECT 和 SELECT * 语句 。 


SQL SELECT 语句 


SELECT 语句 用 于 从 表 中 选取 数据 。 
结果 被 存储 在 一 个 结果 表 中 ( 称 为 结果 集 ) 。 


SQL SELECT 语法 
SELECT 列 名称 FROM RAF 
以 及 : 


SELECT * FROM 表 名 称 
注释 : SQL 语句 对 大 小 写 不 敏感 。SELECT SRF select. 


SQL SELECT 实例 


如 需 获 取 名 为 "LastName" 和 "FirstName" 的 列 的 内 容 (从 名 为 "Persons" 的 数据 
库 表 ) ， 请 使 用 类 似 这 样 的 SELECT 语句 : 


SELECT LastName, FirstName FROM Persons 


"Persons" 3: 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


结果 : 


LastName FirstName 


Adams John 
Bush George 
Carter Thomas 


SQL SELECT * 实例 


现在 我 们 希望 从 "Persons" 表 中 选取 所 有 的 列 。 
请 使 用 符号 * 取代 列 的 名 称 ， 就 像 这 样 : 


SELECT * FROM Persons 


提示 : £5 (*) 是 选取 所 有 列 的 快捷 方式 。 


ER : 
Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


在 结果 集 (result-set) 中 导航 


由 SQL 查询 程序 获得 的 结果 被 存放 在 一 个 结果 集中 。 大 多 数 数据 库 软 件 系统 都 多 
许 使 用 编程 函数 在 结果 集中 进行 导航 ， 上 比如 : Move-To-First-Record、Get-Record- 
Content、Move-To-Next-Record 等 等 。 


类 似 这 些 编程 画 数 不 在 本 教程 讲解 之 列 。 如 需 学 习 通 过 男 数 调用 访问 数据 的 知识 ， 
请 访问 我 们 的 ADO 教程 和 PHP 教程 。 


SQL SELECT DISTINCT 语句 


本 章 讲 解 SELECT DISTINCT 语句 。 


SQL SELECT DISTINCT j24) 


在 表 中 ， 可 能 会 包含 重复 值 。 这 并 不 成 问题 ， 不 过 ， 有 时 您 也 许 希望 仅仅 列 出 不 同 
(distinct) 的 值 。 
关键 词 DISTINCT 用 于 返回 唯一 不 同 的 值 。 


语法 : 


SELECT DISTINCT 列 名 称 FROM 表 名 称 


使 用 DISTINCT 关键 词 
如 果 要 从 "Company" 列 中 选取 所 有 的 值 ， 我 们 需要 使 用 SELECT 话 句 : 


SELECT Company FROM Orders 


"Orders" 表 : 
Company OrderNumber 
IBM 3532 
W3School 2356 
Apple 4698 
W3School 6953 


BN 


ERR: 


Company 
IBM 
W3School 
Apple 
W3School 
请 注意 ， 在 结果 集中 ，W3School 被 列 出 了 两 次 。 
如 需 从 Company" 列 中 仅 选 取 唯 一 不 同 的 值 ， 我 们 需要 使 用 SELECT DISTINCT 


语句 : 


SELECT DISTINCT Company FROM Orders 


结果 : 


Company 
IBM 
W3School 
Apple 


现在 ， 在 结果 集中 ，"W3School" 仅 被 列 出 了 一 次 。 


SQL WHERE 子 句 
WHERE 子 句 用 于 规定 选择 的 标准 。 
WHERE 子 句 
如 需 有 条 件 地 从 表 中 选取 数据 ， 可 将 WHERE 子 句 添加 到 SELECT 语句 。 
语法 
SELECT 列 名 称 FROM 表 名 称 WHERE 列 运算 符 值 


下 面 的 运算 符 可 在 WHERE 子 句 中 使 用 : 


操作 符 描述 
= 等 于 
<> RSF 
> AF 
< INF 
a Ria 
<= 小 于 等 于 
BETWEEN 在 某 个 范围 内 
LIKE 搜索 某 种 模式 


注释 : 在 某 些 版 本 的 SQL 中 ， 操 作 符 <> 可 以 写 为 !=。 


使 用 WHERE +4) 


如 果 只 希望 选取 居住 在 城市 "Beijing" 中 的 人 ， 我 们 需要 向 SELECT 语句 添加 
WHERE 子 句 : 


SELECT * FROM Persons WHERE City='Beijing' 


"Persons" 表 


LastName 
Adams 


Bush 


Any 
ty 
A 


LastName 
Carter 


Gates 


引号 的 使 用 


FirstName 
John 
George 
Thomas 


Bill 


FirstName 
Thomas 


Bill 


Address 
Oxford Street 
Fifth Avenue 
Changan Street 


Xuanwumen 10 


Address 
Changan Street 


Xuanwumen 10 


请 注意 ， 我 们 在 例子 中 的 条 件 值 周围 使 用 的 是 单 引 号 。 
SQL 使 用 单 引 号 来 环绕 文本 值 (大 部 分 数据 库 系统 也 接受 双 引 号 ) 。 如 果 是 数值 ， 


请 不 要 使 用 引号 。 


文本 值 : 


这 是 正确 的 : 


SELECT * FROM Persons WHERE FirstName='Bush' 


这 是 错误 的 : 


SELECT * FROM Persons WHERE FirstName=Bush 


数值 : 


这 是 正确 的 : 


SELECT * FROM Persons WHERE Year>1965 


这 是 错 误 的 : 


SELECT * FROM Persons WHERE Year>'1965' 


City 
London 
New York 
Beijing 
Beijing 


City 
Beijing 
Beijing 


Year 
1970 
1975 
1980 
1985 


Year 
1980 
1985 


SQL AND & OR 运算 符 
AND 和 OR 运算 符 用 于 基于 一 个 以 上 的 条 件 对 记录 进行 过 滤 。 


AND 和 OR 运算 符 


AND 和 OR 可 在 WHERE 子 语句 中 把 两 个 或 多 个 条 件 结合 起 来 。 
如 果 第 一 个 条 件 和 第 二 个 条 件 都 成 立 ， 则 AN 运算 符 显 示 一 条 记录 。 
如 果 第 一 个 条 件 和 第 二 个 条 件 中 只 要 有 一 个 成 立 ， 则 OR 运算 符 显示 一 条 记录 。 


原始 的 表 (用 在 例子 中 的 ) : 


LastName FirstName Address City 
Adams John Oxford Street London 
Bush George Fifth Avenue New York 
Carter Thomas Changan Street Beijing 
Carter William Xuanwumen 10 Beijing 


AND 运算 符 实例 
使 用 AND 来 显示 所 有 姓 为 "Carter" 并 且 名 为 "Thomas" 的 人 : 


SELECT * FROM Persons WHERE FirstName='Thomas' AND LastName='Cartel 





结 来 : 
LastName FirstName Address City 
Carter Thomas Changan Street Beijing 
OR 运算 符 实例 


使 用 OR 来 显示 所 有 姓 为 "Carter" REZ A "Thomas" 的 人 : 


SELECT * FROM Persons WHERE firstname='Thomas' OR lastname='Carter 








te: 
LastName FirstName Address City 
Carter Thomas Changan Street Beijing 
Carter William Xuanwumen 10 Beijing 


结合 AND 和 OR 运算 符 
我 们 也 可 以 把 AND 和 OR 结合 起 来 〈 使 用 圆 括号 来 组 成 复杂 的 表达 式 ) : 


SELECT * FROM Persons WHERE (FirstName='Thomas' OR FirstName='Will: 
AND LastName='Carter' 





LastName FirstName Address City 
Carter Thomas Changan Street Beijing 


Carter William Xuanwumen 10 Beijing 


SQL ORDER BY 子 句 
ORDER BY 语句 用 于 对 结果 集 进 行 排序 。 


ORDER BY 语句 


ORDER BY 语句 用 于 根据 指定 的 列 对 结果 集 进 行 
ORDER BY 语句 默认 按照 升序 对 记录 进行 排序 。 
如 果 您 希望 按照 降序 对 记录 进行 排序 ， 可 以 使 用 DESC 关键 字 。 


原始 的 表 (用 在 例子 中 的 ) : 


排序 。 


Orders X: 
Company OrderNumber 
IBM 3532 
W3School 2356 
Apple 4698 
W3School 6953 
实例 1 


以 字母 顺序 显示 公司 名 称 : 


SELECT Company, OrderNumber FROM Orders ORDER BY Company 


ER : 
Company OrderNumber 
Apple 4698 
IBM 3532 
W3School 6953 


W3School 2356 


实例 2 


以 字母 顺序 显示 公司 名 称 (Company) ， 并 以 数字 顺序 显示 顺序 号 
(OrderNumber) 


SELECT Company, OrderNumber FROM Orders ORDER BY Company, OrderNumt 








4 = ee | 
结果 : 
Company OrderNumber 

Apple 4698 

IBM 3532 

W3School 2356 

W3School 6953 
实例 3 


以 逆 字 母 顺 序 显 示 公 司 名 称 : 


SELECT Company, OrderNumber FROM Orders ORDER BY Company DESC 


结 来 : 
Company OrderNumber 
W3School 6953 
W3School 2356 
IBM 3532 
Apple 4698 
实例 4 


以 逆 字 母 顺 序 显 示 公 司 名 称 ， 并 以 数字 顺序 显示 顺序 号 : 


SELECT Company, OrderNumber FROM Orders ORDER BY Company DESC, Orde 
«| = — g 








结果 : 


Company OrderNumber 
W3School 2356 
W3School 6953 
IBM 3532 
Apple 4698 


注意 : 在 以 上 的 结果 中 有 两 个 相等 的 公司 名 称 (W3School)。 只 有 这 一 次 ， 在 第 一 
列 中 有 相同 的 值 时 ， 第 二 列 是 以 升序 排列 的 。 如 果 第 一 列 中 有 些 值 为 nulls 时 ， 情 
况 也 是 这 样 的 。 


SQL INSERT INTO 语句 


INSERT INTO 语句 


INSERT INTO 语句 用 于 向 表格 中 插入 新 的 行 。 
语法 
INSERT INTO 表 名 称 VALUES ( 值 1， 值 2,....) 


我 们 也 可 以 指定 所 要 插入 数据 的 列 : 


INSERT INTO table_name ( 列 1， 列 2, ...) VALUES ( 值 1， 值 2,....) 


插入 新 的 行 


"Persons" X : 


LastName FirstName Address City 
Carter Thomas Changan Street Beijing 
SQL 语句 : 


INSERT INTO Persons VALUES ('Gates', 'Bill', 'Xuanwumen 10', 'Beij: 





结果 : 
LastName FirstName Address City 
Carter Thomas Changan Street Beijing 
Gates Bill Xuanwumen 10 Beijing 


在 指定 的 列 中 插入 数据 


"Persons" X : 


LastName FirstName Address 
Carter Thomas Changan Street 
Gates Bill Xuanwumen 10 

SQL 语句 : 


City 
Beijing 
Beijing 


INSERT INTO Persons (LastName, Address) VALUES ('Wilson', 'Champs-f 





结果 : 
LastName FirstName Address 
Carter Thomas Changan Street 
Gates Bill Xuanwumen 10 


Wilson Champs-Elysees 


City 
Beijing 
Beijing 


SQL UPDATE 324) 
Update 语句 

Update 语句 用 于 修改 表 中 的 数据 。 
BiB : 


UPDATE 表 名 称 SET 列 名 称 = 新 值 WHERE 列 名 称 = 某 值 


Person: 
LastName FirstName Address City 
Gates Bill Xuanwumen 10 Beijing 
Wilson Champs-Elysees 


更 新 某 一 行 中 的 一 个 列 
我 们 为 lastname = "Wilson" 的 人 添加 firstname : 


UPDATE Person SET FirstName = 'Fred' WHERE LastName = 'Wilson' 


结 来 : 
LastName FirstName Address City 
Gates Bill Xuanwumen 10 Beijing 
Wilson Fred Champs-Elysees 


更 新 某 一 行 中 的 耕 干 列 
我 们 会 修改 地 址 (address) ， 并 添加 城市 名 称 (city) 


UPDATE Person SET Address = 
WHERE LastName = 'Wilson' 


结果 : 
LastName FirstName 
Gates Bill 
Wilson Fred 


"Zhongshan 23', City = 'Nanjing' 


Address 
Xuanwumen 10 


Zhongshan 23 


City 
Beijing 
Nanjing 


SQL DELETE 语句 


DELETE 语句 


DELETE 语句 用 于 删除 表 中 的 行 。 
语法 


DELETE FROM RAEN WHERE 列 名 称 = 值 


Person: 
LastName FirstName Address City 
Gates Bill Xuanwumen 10 Beijing 
Wilson Fred Zhongshan 23 Nanjing 


删除 某 行 
"Fred Wilson" 会 被 删除 : 


DELETE FROM Person WHERE LastName = 'Wilson' 


结果 : 
LastName FirstName Address City 
Gates Bill Xuanwumen 10 Beijing 


删除 所 有 行 


可 以 在 不 删除 表 的 情况 下 删除 所 有 的 行 。 这 意味 着 表 的 结构 、 属 性 和 索引 都 是 完整 
BY : 


DELETE FROM table name 


或 者 : 


DELETE * FROM table_name 


SQL 高 级 


SQL TOP 4) 


TOP 子 句 


TOP 子 句 用 于 规定 要 返回 的 记录 的 数目 。 
对 于 拥有 数 千 条 记录 的 大 型 表 来 说 ，TOP 子 句 是 非常 有 用 的 。 
注释 : 并 非 所 有 的 数据 库 系统 都 支持 TOP FA. 


SQL Server 的 语法 : 


SELECT TOP number|percent column_name(s) 
FROM table_name 


MySQL 和 Oracle 中 的 SQL SELECT TOP 是 等 价 的 
MySQL 语法 


SELECT column_name(s) 
FROM table_name 
LIMIT number 


BF 


SELECT == 
FROM Persons 
LIMIT 5 


Oracle 语法 


SELECT column_name(s) 
FROM table_name 
WHERE ROWNUM <= number 


例子 


SELECT * 
FROM Persons 
WHERE ROWNUM <= 5 


原始 的 表 (用 在 例子 中 的 ) : 


Persons 表 : 
Id LastName FirstName Address 
1 Adams John Oxford Street 
2 Bush George Fifth Avenue 
3 Carter Thomas Changan Street 
4 Obama Barack Pennsylvania Avenue 


SQL TOP 实例 


现在 ， 我 们 希望 从 上 面 的 "Persons'" 表 中 选取 头 两 条 记录 。 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT TOP 2 * FROM Persons 


结果 : 
Id LastName FirstName Address 
1 Adams John Oxford Street 
2 Bush George Fifth Avenue 


SQL TOP PERCENT 实例 


现在 ， 我 们 希望 从 上 面 的 "Persons'" 表 中 选取 50% 的 记录 。 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT TOP 50 PERCENT * FROM Persons 


结果 : 


City 
London 
New York 
Beijing 
Washington 


City 
London 


New York 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 


SQL LIKE 操作 符 


LIKE 操作 符 用 于 在 WHERE 子 句 中 搜索 列 中 的 指定 模式 。 
LIKE 操作 符 

LIKE 操作 符 用 于 在 WHERE 子 句 中 搜索 列 中 的 指定 模式 。 
SQL LIKE 操作 符 语法 


SELECT column_name(s) 
FROM table name 
WHERE column_name LIKE pattern 


原始 的 表 (用 在 例子 中 的 ) : 


Persons 表 : 
Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


LIKE 操作 符 实 例 


例子 1 


现在 ， 我 们 希望 从 上 面 的 "Persons'" 表 中 选取 居住 在 以 "N" 开始 的 城市 里 的 人 : 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City LIKE 'N%' 


提示 : "%" 可 用 于 定义 通配符 (模式 中 缺少 的 字母 ) 。 


Id LastName FirstName Address City 
2 Bush George Fifth Avenue New York 


例子 2 
接 下 来 ， 我 们 希望 从 "Persons" 表 中 选取 居住 在 以 "g" 结尾 的 城市 里 的 人 : 
我 们 可 以 使 用 下 面 的 SELECT 话 句 : 


SELECT * FROM Persons 
WHERE City LIKE '%g' 


结果 集 
Id LastName FirstName Address City 
3 Carter Thomas Changan Street Beijing 
例子 3 


接 下 来 ， 我 们 希望 从 "Persons'" 表 中 选取 居住 在 包含 "lon" 的 城市 里 的 人 : 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City LIKE '%lon%' 


结果 集 
Id LastName FirstName Address City 
1 Adams John Oxford Street London 
例子 4 


通过 使 用 NOT 关键 字 ， 我 们 可 以 从 "Persons'" 表 中 选取 居住 在 不 包含 "lon" 的 城市 
里 的 人 : 


我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City NOT LIKE '%lon%' 


3 


LastName 
Bush 
Carter 


FirstName 
George 


Thomas 


Address 
Fifth Avenue 
Changan Street 


City 
New York 
Beijing 


SQL 通配符 


在 搜索 数据 库 中 的 数据 时 ， 您 可 以 使 用 SQL 通配符 。 


SQL 通配符 

在 搜索 数据 库 中 的 数据 时 ，SQL 通配符 可 以 替代 一 个 或 多 个 字符 。 
SQL 通配符 必须 与 LIKE 运算 符 一 起 使 用 。 

在 SQL 中 ， 可 使 用 以 下 通配符 : 


通配符 描述 
% 替代 一 个 或 多 个 字符 
= 仅 替 代 一 个 字符 
[charlist] 字符 列 中 的 任何 单一 字符 
[Acharlist] 或 者 [!charlist] 不 在 字符 列 中 的 任何 单一 字符 


原始 的 表 (用 在 例子 中 的 ) : Persons X: 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


使 用 % 通配符 


例子 1 
现在 ， 我 们 希望 从 上 面 的 "Persons'" 表 中 选取 居住 在 以 "Ne" 开始 的 城市 里 的 人 : 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City LIKE 'Ne%' 


Id LastName FirstName Address City 
2 Bush George Fifth Avenue New York 


例子 2 
接 下 来 ， 我 们 希望 从 "Persons'" 表 中 选取 居住 在 包含 "lond" 的 城市 里 的 人 : 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City LIKE '%lond%' 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 


使 用 _ 通配符 


例子 1 
现在 ， 我 们 希望 从 上 面 的 "Persons" 表 中 选取 名 字 的 第 一 个 字符 之 后 是 "eorge" 的 
人 : 


我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE FirstName LIKE '_eorge' 


Id LastName FirstName Address City 
2 Bush George Fifth Avenue New York 


例子 2 


接 下 来 ， 我 们 希望 从 "Persons" 表 中 选取 的 这 条 记录 的 姓氏 以 "C" 开头 ， 然 后 是 一 
个 任意 字符 ， 然 后 是 "r"， 然 后 是 任意 字符 ， 然 后 是 "er" : 


我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE LastName LIKE 'C_r_er' 


Id LastName FirstName Address City 
3 Carter Thomas Changan Street Beijing 


使 用 [charlist] 通配符 


例子 1 


现在 ， 我 们 希望 从 上 面 的 "Persons'" 表 中 选取 居住 的 城市 以 "A" BK "L" =% "N" 开头 
的 人 : 


我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City LIKE '[ALN]%' 


结果 集 
Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
例子 2 


现在 ， 我 们 希望 从 上 面 的 "Persons" 表 中 选取 居住 的 城市 不 以 "A" BK "L" BK "N" FF 
头 的 人 : 


我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE City LIKE '[!ALN]%' 


LastName FirstName Address City 


Carter Thomas Changan Street Beijing 


SQL IN 操作 符 
IN 操作 符 


IN 操作 符 允 许 我 们 在 WHERE 子 句 中 规定 多 个 值 。 


SQL IN 语法 


SELECT column_name(s) 
FROM table_name 
WHERE column_name IN (value1,value2,...) 


原始 的 表 (在 实例 中 使 用 : ) 


Persons 表 : 
Id LastName FirstName Address 
1 Adams John Oxford Street 
2 Bush George Fifth Avenue 
3 Carter Thomas Changan Street 


IN 操作 符 实 例 


现在 ， 我 们 希望 从 上 表 中 选取 姓氏 为 Adams 和 Carter 的 人 : 
我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT * FROM Persons 
WHERE LastName IN ('Adams', 'Carter') 


结果 集 
Id LastName FirstName Address 
1 Adams John Oxford Street 
3 Carter Thomas Changan Street 


City 
London 
New York 
Beijing 


City 
London 


Beijing 


SQL BETWEEN 操作 符 


BETWEEN 操作 符 在 WHERE 子 句 中 使 用 ， 作 用 是 选取 介 于 两 个 值 之 间 的 数据 范 
围 。 


BETWEEN 操作 符 


操作 符 BETWEEN .… AND 会 选取 介 于 两 个 值 之 间 的 数据 范围 。 这 些 值 可 以 是 数 


SQL BETWEEN 语法 


SELECT column_name(s) 
FROM table_name 

WHERE column_name 

BETWEEN value1 AND Value2 


原始 的 表 (在 实例 中 使 用 : ) 


Persons 表 : 
Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 
4 Gates Bill Xuanwumen 10 Beijing 


BETWEEN 操作 符 实例 


如 需 以 字母 顺序 显示 介 于 "Adams" (包括 ) 和 "Carter”( 不 包括 ) 之 间 的 人 ， 请 使 
用 下 面 的 SQL : 


SELECT * FROM Persons 
WHERE LastName 
BETWEEN 'Adams' AND 'Carter' 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 


重要 事项 : 不 同 的 数据 库 对 BETWEEN...AND 操作 符 的 处 理 方式 是 有 差异 的 。 某 
些 数据 库 会 列 出 介 于 "Adams" 和 "Carter" 之 间 的 人 ， 但 不 包括 "Adams" 和 
"Carter" ; 某 些 数据 库 会 列 出 介 于 "Adams" 和 "Carter" 之 间 并 包括 "Adams" 和 
"Carter" 的 人 ; 而 另 一 些 数据 库 会 列 出 介 于 "Adams" 和 "Carter" Zia AIA, Bis 
"Adams" ， 但 不 包括 "Carter" 。 


所 以 ， 请 检查 你 的 数据 库 是 如 何 处 理 BETWEEN....AND 操作 符 的 ! 


实例 2 
如 需 使 用 上 面 的 例子 显示 范围 之 外 的 人 ， 请 使 用 NOT 操作 符 : 


SELECT * FROM Persons 
WHERE LastName 
NOT BETWEEN 'Adams' AND 'Carter' 


Id LastName FirstName Address City 
3 Carter Thomas Changan Street Beijing 


4 Gates Bill Xuanwumen 10 Beijing 


SQL Alias (别名 ) 


通过 使 用 SQL， 可 以 为 列 名 称 和 表 名 称 指定 别名 (Alias) 。 
SQL Alias 
RAJ SQL Alias 语法 


SELECT column_name(s) 
FROM table_name 
AS alias_name 


列 的 SQL Alias 语 ; 


SELECT column_name AS alias name 
FROM table_name 


Alias 实例 : 使 用 表 名 称 别 名 

假设 我 们 有 两 个 表 分 别 是 : "Persons" 和 "Product_Orders"。 我 们 分 别 为 它们 指定 
别名 "p" 和 "po"。 

现在 ， 我 们 希望 列 出 "John Adams" 的 所 有 定单 。 

我 们 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT po.OrderID, p.LastName, p.FirstName 
FROM Persons AS p, Product_Orders AS po 
WHERE p.LastName='Adams' AND p.FirstName='John' 


不 使 用 别名 的 SELECT 语句 : 


SELECT Product_Orders.OrderID, Persons.LastName, Persons.FirstName 
FROM Persons, Product_Orders 
WHERE Persons.LastName='Adams' AND Persons.FirstName='John' 


4 a L 
NER AR SELECT 语句 您 可 以 看 到 ， 别 名 使 查询 程序 更 易 阅 读 和 书写 。 








Alias 实例 : 使 用 一 个 列 名 别名 


表 Persons: 
Id LastName FirstName Address 
1 Adams John Oxford Street 
2 Bush George Fifth Avenue 
3 Carter Thomas Changan Street 
SQL: 


SELECT LastName AS Family, FirstName AS Name 
FROM Persons 


结果 : 
Family Name 
Adams John 
Bush George 


Carter Thomas 


City 
London 
New York 
Beijing 


SQL JOIN 


SQL join 用 于 根据 两 个 或 多 个 表 中 的 列 之 间 的 关系 ， 从 这 些 表 中 查询 数据 。 


Join 和 Key 
有 时 为 了 得 到 完整 的 结果 ， 我 们 需要 从 两 个 或 更 多 的 表 中 获取 结果 。 我 们 就 需要 执 
行 join。 


数据 库 中 的 表 可 通过 键 将 彼此 联系 起 来 。 主 键 (Primary Key) 是 一 个 列 ， 在 这 个 
列 中 的 每 一 行 的 值 都 是 唯一 的 。 在 表 中 ， 每 个 主键 的 值 都 是 唯一 的 。 这 样 做 的 目的 
是 在 不 重复 每 个 表 中 的 所 有 数据 的 情况 下 ， 把 表 间 的 数据 交叉 捆绑 在 一 起 。 


请 看 "Persons" K : 


Id_P LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


请 注意 ，"ld_P" 列 是 Persons 表 中 的 的 主键 。 这 意味 着 没有 两 行 能 够 拥有 相同 的 
lId_P。 即 使 两 个 人 的 姓名 完全 相同 ，Id_P 也 可 以 区 分 他 们 。 


接 下 来 请 看 "Orders" X : 


Id_O OrderNo Id_P 
1 77895 3 
2 44678 3 
3 22456 1 
4 24562 1 
5 34764 65 


请 注意 ，"ld_O" 列 是 Orders 表 中 的 的 主键 ， 同 时 ，"Orders" 表 中 的 "Id_P" 列 用 于 
引用 "Persons" 表 中 的 人 ， 而 无 需 使 用 他 们 的 确切 姓名 。 


请 留意 ，"ld_P" 列 把 上 面 的 两 个 表 联 系 了 起 来 。 


引用 两 个 表 


我 们 可 以 通过 引用 两 个 表 的 方式 ， 从 两 个 表 中 获取 数据 : 
谁 订 购 了 产品 ， 并 且 他 们 订购 了 什么 产品 ? 
SELECT Persons.LastName, Persons.FirstName, Orders.OrderNo 


FROM Persons, Orders 
WHERE Persons.Id_P = Orders.Id_P 


结果 集 
LastName FirstName OrderNo 
Adams John 22456 
Adams John 24562 
Carter Thomas 77895 
Carter Thomas 44678 


SQL JOIN - 使 用 Join 


除了 上 面 的 方法 ， 我 们 也 可 以 使 用 关键 词 JOIN 来 从 两 个 表 中 获取 数据 。 
如 果 我 们 希望 列 出 所 有 人 的 定购 ， 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT Persons.LastName, Persons.FirstName, Orders.OrderNo 
FROM Persons 

INNER JOIN Orders 

ON Persons.Id_P = Orders.Id_P 

ORDER BY Persons.LastName 


结果 集 
LastName FirstName OrderNo 
Adams John 22456 
Adams John 24562 
Carter Thomas 77895 
Carter Thomas 44678 


不 同 的 SQL JOIN 


除了 我 们 在 上 面 的 例子 中 使 用 的 INNER JOIN (内 连接 ) ， 我 们 还 可 以 使 用 其 他 几 
种 连接 。 


下 面 列 出 了 您 可 以 使 用 的 JOIN 类 型 ， 以 及 它们 之 间 的 差异 。 


e JOIN: 如 果 表 中 有 至 少 一 个 匹配 ， 则 返回 行 

e LEFT JOIN: 即使 右 表 中 没有 匹配 ， 也 从 左 表 返回 所 有 的 行 
e RIGHT JOIN: 即使 左 表 中 没有 匹配 ， 也 从 右 表 返回 所 有 的 行 
e FULL JOIN: 只 要 其 中 一 个 表 中 存在 匹配 ， 就 返回 行 


SQL INNER JOIN 关键 字 


SQL INNER JOIN 关键 字 


在 表 中 存在 至 少 一 个 匹配 时 ，INNER JOIN 关键 字 返 回 行 。 


INNER JOIN 关键 字 语 法 


SELECT column_name(s) 

FROM table namel1 

INNER JOIN table name2 

ON table_name1.column_name=table_name2.column_name 


注释 : INNER JOIN 与 JOIN 是 相同 的 。 


原始 的 表 (用 在 例子 中 的 ) : 


"Persons" 3 : 
Id_P LastName FirstName Address 
1 Adams John Oxford Street 
2 Bush George Fifth Avenue 
3 Carter Thomas Changan Street 
"Orders" X : 
Id_O OrderNo 
1 77895 3 
2 44678 3 
3 22456 1 
4 24562 1 
5 34764 65 


内 连接 (INNER JOIN) 实例 


City 
London 
New York 
Beijing 


现在 ， 我 们 希望 列 出 所 有 人 的 定购 。 
您 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT Persons.LastName, Persons.FirstName, Orders.OrderNo 
FROM Persons 

INNER JOIN Orders 

ON Persons.Id_P=Orders.Id_P 

ORDER BY Persons.LastName 


结果 集 
LastName FirstName OrderNo 
Adams John 22456 
Adams John 24562 
Carter Thomas 77895 
Carter Thomas 44678 


INNER JOIN 关键 字 在 表 中 存在 至 少 一 个 匹配 时 返回 行 。 如 果 "Persons" 中 的 行 在 
"Orders" 中 没有 匹配 ， 就 不 会 列 出 这 些 行 。 


SQL LEFT JOIN 关键 字 


SQL LEFT JOIN 关键 字 


LEFT JOIN 关键 字 会 从 左 表 (table_name1) 那里 返回 所 有 的 行 ， 即 使 在 右 表 
(table_name2) 中 没有 匹配 的 行 。 


LEFT JOIN 关键 字 语 法 


SELECT column_name(s) 

FROM table namel1 

LEFT JOIN table_name2 

ON table_namei.column_name=table_name2.column_name 


注释 : 在 某 些 数据 库 中 ， LEFT JOIN 称 为 LEFT OUTER JOIN. 


原始 的 表 (用 在 例子 中 的 ) : 


"Persons" 表 : 
Id_P LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 
"Orders" X : 
Id_O OrderNo Id_P 
1 77895 3 
2 44678 3 
3 22456 1 
4 24562 1 
5 34764 65 


左 连接 (LEFT JOIN) 实例 


现在 ， 我 们 希望 列 出 所 有 的 人 ， 以 及 他 们 的 定购 - 如 果 有 的 话 。 


您 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT Persons.LastName, Persons.FirstName, Orders.OrderNo 


FROM Persons 

LEFT JOIN Orders 

ON Persons.Id_P=Orders.Id_P 
ORDER BY Persons.LastName 


结果 集 
LastName FirstName 
Adams John 
Adams John 
Carter Thomas 
Carter Thomas 
Bush George 


OrderNo 
22456 
24562 
77895 
44678 


LEFT JOIN 关键 字 会 从 左 表 (Persons) 那里 返回 所 有 的 行 ， 即 使 在 右 表 (Orders) 


中 没有 匹配 的 行 。 


SQL RIGHT JOIN 关键 字 


SQL RIGHT JOIN 关键 字 


RIGHT JOIN 关键 字 会 右 表 (table_name2) 那里 返回 所 有 的 行 ， 即 使 在 左 表 


(table_name1) 中 没有 匹配 的 行 。 


RIGHT JOIN 关键 字 语 法 


SELECT column_name(s) 
FROM table_name1 
RIGHT JOIN table_name2 


ON table_name1.column_name=table_name2.column_name 


注释 : 在 某 些 数据 库 中 ， RIGHT JOIN 称 为 RIGHT OUTER JOIN. 


原始 的 表 (用 在 例子 中 的 ) : 


Oxford Street 
Fifth Avenue 


Changan Street 


"Persons" X : 
Id_P LastName FirstName 

1 Adams John 

2 Bush George 

3 Carter Thomas 
"Orders" X : 

Id_O OrderNo 

1 77895 

2 44678 

3 22456 

4 24562 

5 34764 


右 连 接 (RIGHT JOIN) 实例 


65 


City 
London 
New York 
Beijing 


现在 ， 我 们 希望 列 出 所 有 的 定单 ， 以 及 定购 它们 的 人 - 如 果 有 的 话 。 
您 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT Persons.LastName, 


FROM Persons 


RIGHT 


JOIN Orders 


ON Persons.Id P=Orders.Id P 


ORDER 


Adams 
Adams 
Carter 
Carter 
34764 


BY Persons.LastName 
LastName FirstName 
John 
John 
Thomas 
Thomas 


Persons.FirstName, Orders.OrderNo 


OrderNo 
22456 
24562 
77895 
44678 


RIGHT JOIN 关键 字 会 从 右 表 (Orders) 那里 返回 所 有 的 行 ， 即 使 在 左 表 (Persons) 
中 没有 匹配 的 行 。 


SQL FULL JOIN 关键 字 


SQL FULL JOIN 关键 字 


只 要 其 中 某 个 表 存 在 匹配 ，FULL JOIN 关键 字 就 会 返回 行 。 


FULL JOIN 关键 字 话 法 


SELECT column_name(s) 

FROM table namel1 

FULL JOIN table_name2 

ON table_name1.column_name=table_name2.column_name 


注释 : 在 某 些 数据 库 中 ， FULL JOIN 称 为 FULL OUTER JOIN. 


原始 的 表 (用 在 例子 中 的 ) : 


"Persons" 表 : 
Id_P LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 
"Orders" X : 
Id_O OrderNo Id_P 
1 77895 3 
2 44678 3 
3 22456 1 
4 24562 1 
5 34764 65 


全 连接 (FULL JOIN) 实例 


现在 ， 我 们 希望 列 出 所 有 的 人 ， 以 及 他 们 的 定单 ， 以 及 所 有 的 定单 ， 以 及 定购 它们 
的 人 。 


您 可 以 使 用 下 面 的 SELECT 语句 : 


SELECT Persons.LastName, Persons.FirstName, Orders.OrderNo 
FROM Persons 

FULL JOIN Orders 

ON Persons.Id_P=Orders.Id_P 

ORDER BY Persons.LastName 


结果 集 
LastName FirstName OrderNo 

Adams John 22456 

Adams John 24562 

Carter Thomas 77895 

Carter Thomas 44678 

Bush George 

34764 


FULL JOIN 关键 字 会 从 左 表 (Persons) HAR (Orders) 那里 返回 所 有 的 行 。 如 果 
"Persons" 中 的 行 在 表 "Orders" 中 没有 匹配 ， 或 者 如 果 "Orders" HATER 
"Persons" 中 没有 匹配 ， 这 些 行 同样 会 列 出 。 


SQL UNION 和 UNION ALL 操作 符 


SQL UNION 操作 符 


UNION 操作 符 用 于 合并 两 个 或 多 个 SELECT 语句 的 结果 集 。 


请 注意 ，UNION 内 部 的 SELECT 语句 必须 拥有 相同 数量 的 列 。 列 也 必须 拥有 相似 
的 数据 类 型 。 同 时 ， 每 条 SELECT 语句 中 的 列 的 顺序 必须 相同 。 


SQL UNION 语法 


SELECT column_name(s) FROM table namel1 
UNION 
SELECT column_name(s) FROM table_name2 


注释 : 默认 地 ，UNION 操作 符 选 取 不 同 的 值 。 如 果 人 允许 重复 的 值 ， 请 使 用 UNION 
ALL。 


SQL UNION ALL 语法 


SELECT column_name(s) FROM table namel1 
UNION ALL 
SELECT column_name(s) FROM table_name2 


BA, UNION 结果 集中 的 列 名 总 是 等 于 UNION 中 第 一 个 SELECT 语句 中 的 列 


o 


下 面 的 例子 中 使 用 的 原始 表 : 


Employees_ China: 


E ID E Name 
01 Zhang, Hua 
02 Wang, Wei 
03 Carter, Thomas 


04 Yang, Ming 


Employees_USA: 


E ID E Name 
01 Adams, John 
02 Bush, George 
03 Carter, Thomas 
04 Gates, Bill 


使 用 UNION 命 兮 


实例 
列 出 所 有 在 中 国 和 美国 的 不 同 的 履 员 名 : 
SELECT E_Name FROM Employees_China 


UNION 
SELECT E_Name FROM Employees_USA 


25 FR 


E Name 
Zhang, Hua 
Wang, Wei 
Carter, Thomas 
Yang, Ming 
Adams, John 
Bush, George 
Gates, Bill 
注释 : 这 个 命令 无 法 列 出 在 中 国 和 美国 的 所 有 历 员 。 在 上 面 的 例子 中 ， 我 们 有 两 个 


名 字 相 同 的 履 员 ， 他 们 当中 只 有 一 个 人 被 列 出 来 了 。UNION 命令 只 会 选取 不 同 的 
值 。 


UNION ALL 


UNION ALL 命令 和 UNION 命令 几乎 是 等 效 的 ， 不 过 UNION ALL 命令 会 列 出 所 有 
的 值 。 


SQL Statement 1 
UNION ALL 
SQL Statement 2 


使 用 UNION ALL 命 兮 


实例 : 
列 出 在 中 国 和 美国 的 所 有 的 履 员 : 
SELECT E_Name FROM Employees_China 


UNION ALL 
SELECT E_Name FROM Employees_USA 


25 FR 


E Name 

Zhang, Hua 

Wang, Wei 

Carter, Thomas 

Yang, Ming 

Adams, John 

Bush, George 

Carter, Thomas 


Gates, Bill 


SQL SELECT INTO 语句 


SQL SELECT INTO 语句 可 用 于 创建 表 的 各 份 复 件 。 


SELECT INTO j24) 

SELECT INTO 语句 从 一 个 表 中 选取 数据 ， 然 后 把 数据 插入 另 一 个 表 中 。 
SELECT INTO 语句 常用 于 创建 表 的 备份 复 件 或 者 用 于 对 记录 进行 存档 。 
SQL SELECT INTO 活 法 

您 可 以 把 所 有 的 列 插入 新 表 : 


SELECT * 
INTO new_table_ name [IN externaldatabase] 
FROM old_tablename 


或 者 只 把 希望 的 列 插入 新 表 : 


SELECT column_name(s) 
INTO new_table name [IN externaldatabase ] 
FROM old_tablename 


SQL SELECT INTO 实例 - 制作 各 份 复 件 
下 面 的 例子 会 制作 "Persons'" 表 的 各 份 复 件 : 


SELECT * 
INTO Persons_backup 
FROM Persons 


IN 子 句 可 用 于 向 另 一 个 数据 库 中 拷贝 表 : 


SELEC AS 
INTO Persons IN 'Backup.mdb' 
FROM Persons 


如 果 我 们 希望 拷贝 某 些 域 ， 可 以 在 SELECT 语句 后 列 出 这 些 域 : 


SELECT LastName, FirstName 
INTO Persons_backup 
FROM Persons 


SQL SELECT INTO 实例 - 带 有 WHERE £4) 


我 们 也 可 以 添加 WHERE #4). 


下 面 的 例子 通过 从 "Persons" 表 中 提取 居住 在 "Beijing" 的 人 的 信息 ， 创 建 了 一 个 带 
有 两 个 列 的 名 为 "Persons_backup" HR : 


SELECT LastName,Firstname 
INTO Persons_backup 

FROM Persons 

WHERE City='Beijing' 


SQL SELECT INTO 实例 - 被 连接 的 表 


从 一 个 以 上 的 表 中 选取 数据 也 是 可 以 做 到 的 。 


下 面 的 例子 会 创建 一 个 名 为 "Persons_Order_Backup" 的 新 表 ， 其 中 包含 了 从 
Persons 和 Orders 两 个 表 中 取得 的 信息 : 


SELECT Persons.LastName, Orders.OrderNo 
INTO Persons_Order_Backup 

FROM Persons 

INNER JOIN Orders 

ON Persons.Id_P=Orders.Id_P 


SQL CREATE DATABASE 语句 


CREATE DATABASE 语句 


CREATE DATABASE 用 于 创建 数据 库 。 


SQL CREATE DATABASE 语法 


CREATE DATABASE database_name 


SQL CREATE DATABASE 实例 


现在 我 们 希望 创建 一 个 名 为 "my_db" 的 数据 库 。 
我 们 使 用 下 面 的 CREATE DATABASE 语句 : 


CREATE DATABASE my_db 


可 以 通过 CREATE TABLE 来 添加 数据 库 表 。 


SQL CREATE TABLE 语句 


CREATE TABLE 语句 


CREATE TABLE 语句 用 于 创建 数据 库 中 的 表 。 


SQL CREATE TABLE 语法 


CREATE TABLE RAF 
( 

列 名 称 1 数据 类 型 ， 

列 名 称 2 数据 类 型 ， 

列 名 称 3 数据 类 型 ， 

) 


数据 类 型 (data_type) 规定 了 列 可 容纳 何 种 数据 类 型 。 下 面 的 表格 包含 了 SQL 中 最 
常用 的 数据 类 型 : 


数据 类 型 描述 
integer(size) int(size) 
smallint(size) 仅 容纳 整数 。 在 括号 内 规定 数字 的 最 大 位 数 。 
tinyint(size) 
decimal(size,d) 容纳 带 有 小 数 的 数字 。 "size" 规定 数字 的 最 大 位 
numeric(size,d) 数 。"d" 规定 小 数 点 右 侧 的 最 大 位 数 。 
char(size) 容纳 固定 长 度 的 字符 串 (可 容纳 字母 、 数 字 以 及 特 

殊 字符 ) 。 在 括号 中 规定 字符 串 的 长 度 。 


容纳 可 变 长 度 的 字符 串 (可 容纳 字母 、 数 字 以 及 特 


varchar (cize) 殊 的 字符 ) 。 在 括号 中 规定 字符 串 的 最 大 长 度 。 
date(yyyymmdd) 容纳 日 期 。 


SQL CREATE TABLE 实例 


本 例 演示 如 何 创 建 名 为 "Person" 的 表 。 


该 表 包 含 5 个 列 ， 列 名 分 别 是 : "dP", "LastName", "FirstName", "Address" 以 
及 "City" . 


CREATE TABLE Persons 


( 

Id_P int, 

LastName varchar(255), 
FirstName varchar(255), 
Address varchar(255), 
City varchar(255) 


) 


Id_P 列 的 数据 类 型 是 int， 包 含 整数 。 其 余 4 列 的 数据 类 型 是 varchar， 最 大 长 度 为 
255 个 字符 。 


空 的 "Persons" 表 类 似 这 样 : 


Id_P LastName FirstName Address City 


可 使 用 INSERT INTO 语句 向 空 表 写 入 数据 。 


SQL 约束 (Constraints) 


SQL 约束 


约束 用 于 限制 加 入 表 的 数据 的 类 型 。 


可 以 在 创建 表 时 规定 约束 (通过 CREATE TABLE #4) ， 或 者 在 表 创 建 之 后 也 可 
以 (通过 ALTER TABLE 语句 ) 。 


我 们 将 主要 探讨 以 下 几 种 约束 : 


NOT NULL 
UNIQUE 
PRIMARY KEY 
FOREIGN KEY 
CHECK 
DEFAULT 


注释 : 在 下 面 的 章节 ， 我 们 会 详细 讲解 每 一 种 约束 。 


SQL NOT NULL 约束 


SQL NOT NULL 约束 


NOT NULL 约束 强制 列 不 接受 NULL 值 。 


NOT NULL 约束 强制 字段 始终 包含 值 。 这 意味 着 ， 如 果 不 向 字段 添加 值 ， 就 无 法 插 
入 新 记录 或 者 更 新 记录 。 


下 面 的 SQL 语句 强制 "Id_P" FW "LastName" 列 不 接受 NULL 值 : 


CREATE TABLE Persons 

( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar (255) 

) 


SQL UNIQUE 约束 


SQL UNIQUE 约束 


UNIQUE 约束 唯一 标识 数据 库 表 中 的 每 条 记录 。 
UNIQUE 和 PRIMARY KEY 约束 均 为 列 或 列 集合 提供 了 唯一 性 的 保证 。 
PRIMARY KEY 拥有 自动 定义 的 UNIQUE 约束 。 


请 注意 ， 每 个 表 可 以 有 多 个 UNIQUE 约束 ， 但 是 每 个 表 只 能 有 一 个 PRIMARY 
KEY 约束 。 


SQL UNIQUE Constraint on CREATE TABLE 


下 面 的 SQL 在 "Persons" 表 创 建 时 在 "Id_P" 列 创 建 UNIQUE 约束 : 
MySQL: 


CREATE TABLE Persons 

( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

UNIQUE (Id_P) 

) 


SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 

( 

Id_P int NOT NULL UNIQUE, 
LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar (255) 

) 


如 果 需 要 命名 UNIQUE 约束 ， 以 及 为 多 个 列 定义 UNIQUE 约束 ， 请 使 用 下 面 的 
SQL 语法 : 


MySQL / SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 


( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 

FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

CONSTRAINT uc_PersonID UNIQUE (Id_P, LastName) 


SQL UNIQUE Constraint on ALTER TABLE 


当 表 已 被 创建 时 ， 如 需 在 "Id_P" 列 创建 UNIQUE 约束 ， 请 使 用 下 列 SQL: 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ADD UNIQUE (Id_P) 


如 需 命名 UNIQUE 约束 ， 并 定义 多 个 列 的 UNIQUE 约束 ， 请 使 用 下 面 的 SQL 语 
法 : 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ADD CONSTRAINT uc_PersonID UNIQUE (Id_P, LastName) 


撤销 UNIQUE 约束 


如 需 撤 销 UNIQUE 约束 ， 请 使 用 下 面 的 SQL: 
MySQL: 


ALTER TABLE Persons 
DROP INDEX uc_PersonID 


SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
DROP CONSTRAINT uc_PersonID 


SQL PRIMARY KEY 约束 


SQL PRIMARY KEY 约束 


PRIMARY KEY 约束 唯一 标识 数据 库 表 中 的 每 条 记录 。 
主键 必须 包含 唯一 的 值 。 

主键 列 不 能 包含 NULL 值 。 

每 个 表 都 应 该 有 一 个 主键 ， 并 且 每 个 表 只 能 有 一 个 主键 。 


SQL PRIMARY KEY Constraint on CREATE TABLE 


下 面 的 SQL 4 "Persons" 表 创 建 时 在 "ld_P" 列 创建 PRIMARY KEY 约束 : 
MySQL: 


CREATE TABLE Persons 

( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

PRIMARY KEY (Id_P) 


SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 

( 

Id_P int NOT NULL PRIMARY KEY, 
LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar (255) 

) 


如 果 需 要 命名 PRIMARY KEY 约束 ， 以 及 为 多 个 列 定义 PRIMARY KEY 2OR, if 
使 用 下 面 的 SQL 语法 : 


MySQL / SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 


( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 

FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

CONSTRAINT pk_PersonID PRIMARY KEY (Id_P, LastName) 


SQL PRIMARY KEY Constraint on ALTER TABLE 


如 果 在 表 已 存在 的 情况 下 为 "Id_P" 列 创建 PRIMARY KEY 约束 ， 请 使 用 下 面 的 
SQL : 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ADD PRIMARY KEY (Id_P) 


如 果 需 要 命名 PRIMARY KEY 约束 ， 以 及 为 多 个 列 定义 PRIMARY KEY 约束 ， 请 
使 用 下 面 的 SQL 语法 : 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ADD CONSTRAINT pk_PersonID PRIMARY KEY (Id_P, LastName) 


注释 : 如 果 您 使 用 ALTER TABLE 语句 添加 主键 ， 必 须 把 主键 列 声明 为 不 包含 
NULL 值 (在 表 首 次 创建 时 ) 。 


撤销 PRIMARY KEY 约束 
如 需 撤销 PRIMARY KEY 约束 ， 请 使 用 下 面 的 SQL : 


MySQL: 


ALTER TABLE Persons 
DROP PRIMARY KEY 


SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
DROP CONSTRAINT pk_PersonID 


SQL FOREIGN KEY 约束 


SQL FOREIGN KEY 约束 


一 个 表 中 的 FOREIGN KEY 指向 另 一 个 表 中 的 PRIMARY KEY. 
让 我 们 通过 一 个 例子 来 解释 外 键 。 请 看 下 面 两 个 表 : 


"Persons" 表 : 
Id_P LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 
"Orders" X : 
Id_O OrderNo Id_P 
1 77895 3 
2 44678 3 
3 22456 1 
4 24562 1 


请 注意 ，"Orders" 中 的 "Id_P" 列 指向 "Persons'" 表 中 的 "ld_P" 列 。 

"Persons" HAY "Id_P" Fz "Persons" 表 中 的 PRIMARY KEY. 

"Orders" 表 中 的 "Id_P" Fi "Orders" 表 中 的 FOREIGN KEY, 

FOREIGN KEY 约束 用 于 预防 破坏 表 之 间 连 接 的 动作 。 

FOREIGN KEY 约束 也 能 防止 非法 数据 插入 外 键 列 ， 因 为 它 必 须 是 它 指向 的 那个 表 


中 的 值 之 一 。 
SQL FOREIGN KEY Constraint on CREATE TABLE 
下 面 的 SQL 在 "Orders" 表 创 建 时 为 "Id_P" 列 创建 FOREIGN KEY : 


MySQL: 


CREATE TABLE Orders 

( 

Id_O int NOT NULL, 

OrderNo int NOT NULL, 

Id_P int， 

PRIMARY KEY (Id_0)， 

FOREIGN KEY (Id_P) REFERENCES Persons(Id_P) 
) 


SQL Server / Oracle / MS Access: 


CREATE TABLE Orders 

( 

Id_O int NOT NULL PRIMARY KEY, 

OrderNo int NOT NULL, 

Id_P int FOREIGN KEY REFERENCES Persons(Id_P) 
) 


如 果 需 要 命名 FOREIGN KEY 约束 ， 以 及 为 多 个 列 定义 FOREIGN KEY 约束 ， 请 


使 用 下 面 的 SQL 语法 : 
MySQL / SQL Server / Oracle / MS Access: 


CREATE TABLE Orders 

( 

Id_O int NOT NULL, 

OrderNo int NOT NULL, 

Id_P int, 

PRIMARY KEY (Id_0)， 

CONSTRAINT fk_PerOrders FOREIGN KEY (Id_P) 
REFERENCES Persons(Id_P) 

) 


SQL FOREIGN KEY Constraint on ALTER TABLE 


如 果 在 "Orders" 表 已 存在 的 情况 下 为 "ld_P" 列 创建 FOREIGN KEY 约束 ， 请 使 用 


下 面 的 SQL : 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Orders 
ADD FOREIGN KEY (Id_P) 
REFERENCES Persons(Id_P) 


如 果 需 要 命名 FOREIGN KEY 约束 ， 以 及 为 多 个 列 定义 FOREIGN KEY 约束 ， 请 
使 用 下 面 的 SQL 语法 : 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Orders 

ADD CONSTRAINT fk_PerOrders 
FOREIGN KEY (Id_P) 
REFERENCES Persons(Id_P) 


撤销 FOREIGN KEY 约束 


如 需 撤销 FOREIGN KEY 约束 ， 请 使 用 下 面 的 SQL : 
MySQL: 


ALTER TABLE Orders 
DROP FOREIGN KEY fk_PerOrders 


SQL Server / Oracle / MS Access: 


ALTER TABLE Orders 
DROP CONSTRAINT fk_PerOrders 


SQL CHECK 约束 


SQL CHECK 约束 


CHECK 约束 用 于 限制 列 中 的 值 的 范围 。 
如 果 对 单个 列 定义 CHECK 约束 ， 那 么 该 列 只 人 允许 特定 的 值 。 
如 果 对 一 个 表 定 义 CHECK 约束 ， 那 么 此 约束 会 在 特定 的 列 中 对 值 进行 限制 。 


SQL CHECK Constraint on CREATE TABLE 


下 面 的 SQL 在 "Persons'" 表 创 建 时 为 "ld_P" 列 创建 CHECK 约束 。CHECK 约束 
规定 "Id_P" 列 必须 只 包含 大 于 0 的 整数 。 


My SQL: 


CREATE TABLE Persons 

( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

CHECK (Id_P>Q) 

) 


SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 

( 

Id_P int NOT NULL CHECK (Id_P>0), 
LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar (255) 

) 


如 果 需 要 命名 CHECK 约束 ， 以 及 为 多 个 列 定义 CHECK 约束 ， 请 使 用 下 面 的 SQL 


语法 : 


MySQL / SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 


( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 

FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

CONSTRAINT chk_Person CHECK (Id_P>0 AND City='Sandnes' ) 


SQL CHECK Constraint on ALTER TABLE 


如 果 在 表 已 存在 的 情况 下 为 "Ild_P" 列 创 建 CHECK 约束 ， 请 使 用 下 面 的 SQL: 


MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ADD CHECK (Id_P>0) 


如 果 需 要 命名 CHECK 约束 ， 以 及 为 多 个 列 定义 CHECK 约束 ， 请 使 用 下 面 的 SQL 


语法 : 
MySQL / SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ADD CONSTRAINT chk_Person CHECK (Id_P>0 AND City='Sandnes' ) 


撤销 CHECK 约束 
如 需 撤 销 CHECK 约束 ， 请 使 用 下 面 的 SQL: 


SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
DROP CONSTRAINT chk_Person 


MySQL: 


ALTER TABLE Persons 
DROP CHECK chk_Person 


SQL DEFAULT 约束 


SQL DEFAULT 约束 


DEFAULT 约束 用 于 向 列 中 插入 默认 值 。 
如 果 没 有 规定 其 他 的 值 ， 那 么 会 将 默认 值 添加 到 所 有 的 新 记录 。 


SQL DEFAULT Constraint on CREATE TABLE 


下 面 的 SQL 在 "Persons'" 表 创 建 时 为 "City" 列 创建 DEFAULT 约束 : 


My SQL / SQL Server / Oracle / MS Access: 


CREATE TABLE Persons 


( 

Id_P int NOT NULL, 

LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar(255) DEFAULT 'Sandnes' 
) 


通过 使 用 类 似 GETDATE() 2 AEX, DEFAULT 约束 也 可 以 用 于 插入 系统 值 : 


CREATE TABLE Orders 

Id_O int NOT NULL, 

OrderNo int NOT NULL, 

Id_P int, 

OrderDate date DEFAULT GETDATE() 
) 


SQL DEFAULT Constraint on ALTER TABLE 


如 果 在 表 已 存在 的 情况 下 为 "City" 列 创建 DEFAULT 约束 ， 请 使 用 下 面 的 SQL : 


MySQL: 


ALTER TABLE Persons 
ALTER City SET DEFAULT 'SANDNES' 


SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ALTER COLUMN City SET DEFAULT 'SANDNES' 


撤销 DEFAULT 约束 


如 需 撤销 DEFAULT 约束 ， 请 使 用 下 面 的 SQL : 
MySQL: 


ALTER TABLE Persons 
ALTER City DROP DEFAULT 


SQL Server / Oracle / MS Access: 


ALTER TABLE Persons 
ALTER COLUMN City DROP DEFAULT 


SQL CREATE INDEX 语句 


CREATE INDEX 语句 用 于 在 表 中 创建 索引 。 
在 不 读 取 整个 表 的 情况 下 ， 索 引 使 数据 库 应 用 程序 可 以 更 快 地 查找 数据 。 


索引 


您 可 以 在 表 中 创建 索引 ， 以 便 更 加 快速 高 效 地 查询 数据 。 
用 户 无 法 看 到 索引 ， 它 们 只 能 被 用 来 加 速 搜索 /查询 。 


注释 : 更 新 一 个 包含 索引 的 表 需 要 比 更 新 一 个 没有 索引 的 表 更 多 的 时 间 ， 这 是 由 于 
索引 本 身 也 需要 更 新 。 因 此 ， 理 想 的 做 法 是 仅仅 在 常常 被 搜索 的 列 〈 以 及 表 ) 上 面 
创建 索引 。 


SQL CREATE INDEX 语法 
在 表 上 创建 一 个 简单 的 索引 。 人 允许 使 用 重复 的 值 : 


CREATE INDEX index_name 
ON table_name (column_name) 


注释 : "column_name" 规定 需要 索引 的 列 。 
SQL CREATE UNIQUE INDEX 语法 
在 表 上 创建 一 个 唯一 的 索引 。 唯 一 的 索引 意味 着 两 个 行 不 能 拥有 相同 的 索引 值 。 


CREATE UNIQUE INDEX index_name 
ON table name (column_name) 


CREATE INDEX 实例 


本 例会 创建 一 个 简单 的 索引 ， 名 为 "Personlndex"， 在 Person 表 的 LastName 
列 : 


CREATE INDEX PersonIndex 
ON Person (LastName) 


如 果 您 希望 以 降序 索引 某 个 列 中 的 值 ， 您 可 以 在 列 名 称 之 后 添加 保留 字 DESC : 


CREATE INDEX PersonIndex 
ON Person (LastName DESC) 


假如 您 希望 索引 不 止 一 个 列 ， 您 可 以 在 括号 中 列 出 这 些 列 的 名 称 ， 用 逗号 隔 开 : 


CREATE INDEX PersonIndex 
ON Person (LastName, FirstName) 


SQL 撤销 索引 、 表 以 及 数据 库 


通过 使 用 DROP 语句 ， 可 以 轻松 地 删除 索引 、 表 和 数据 库 。 


SQL DROP INDEX 语 名 


我 们 可 以 使 用 DROP INDEX 命令 删除 表格 中 的 索引 。 


用 于 Microsoft SQLJet (以 及 Microsoft Access) 的 语法 : 


DROP INDEX index_name ON table name 


用 于 MS SQL Server 的 语法 : 


DROP INDEX table_name.index_name 


用 于 IBM DB2 和 Oracle 语法 : 
DROP INDEX index_name 


用 于 MySQL 的 语法 : 


ALTER TABLE table name DROP INDEX index_name 


SQL DROP TABLE 324) 
DROP TABLE 语句 用 于 删除 表 ( 表 的 结构 、 属 性 以 及 索引 也 会 被 删除 ) : 


DROP TABLE 表 名 称 


SQL DROP DATABASE 语句 


DROP DATABASE 语句 用 于 删除 数据 库 : 


DROP DATABASE 数据 库 名 称 


SQL TRUNCATE TABLE 语句 


如 果 我 们 仅仅 需要 除去 表 内 的 数据 ， 但 并 不 删除 表 本 身 ， 那 么 我 们 该 如 何 做 呢 ? 
请 使 用 TRUNCATE TABLE 命令 (仅仅 删除 表格 中 的 数据 ) : 


TRUNCATE TABLE 表 名 称 


SQL ALTER TABLE 语句 


ALTER TABLE 语句 

ALTER TABLE 语句 用 于 在 已 有 的 表 中 添加 、 修 改 或 删除 列 。 
SQL ALTER TABLE 语法 

如 需 在 表 中 添加 列 ， 请 使 用 下 列 语法 : 


ALTER TABLE table name 
ADD column_name datatype 


要 删除 表 中 的 列 ， 请 使 用 下 列 语法 : 


ALTER TABLE table name 
DROP COLUMN column_name 


注释 : 某 些 数据 库 系 统 不 允许 这 种 在 数据 库 表 中 删除 列 的 方式 (DROP COLUMN 
column_name)。 


要 改变 表 中 列 的 数据 类 型 ， 请 使 用 下 列 语 法 : 


ALTER TABLE table_name 
ALTER COLUMN column_name datatype 


原始 的 表 (用 在 例子 中 的 ) : 


Persons 表 : 
Id LastName FirstName Address City 
1 Adams John Oxford Street London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


SQL ALTER TABLE 实例 


现在 ， 我 们 希望 在 表 "Persons" 中 添加 一 个 名 为 "Birthday" 的 新 列 。 
我 们 使 用 下 列 SQL 语句 : 


ALTER TABLE Persons 
ADD Birthday date 


请 注意 ， 新 列 "Birthday" 的 类 型 是 date， 可 以 存放 上 日期。 数据 类 型 规定 列 中 可 以 存 
放 的 数据 的 类 型 。 


新 的 "Persons" 表 类 似 这 样 : 


Id LastName FirstName Address City Birthday 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 

3 Carter Thomas Changan Street Beijing 


改变 数据 类 型 实例 


现在 我 们 希望 改变 "Persons" RAH "Birthday" 列 的 数据 类 型 。 
我 们 使 用 下 列 SQL 语句 : 


ALTER TABLE Persons 
ALTER COLUMN Birthday year 


请 注意 ，"Birthday" 列 的 数据 类 型 是 year， 可 以 存放 2 位 或 4 位 格式 的 年 份 。 


DROP COLUMN 实例 
接 下 来 ， 我 们 删除 "Person" HAY "Birthday" 列 : 


ALTER TABLE Person 
DROP COLUMN Birthday 


Persons 表 会 成 为 这 样 : 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


SQL AUTO INCREMENT 字段 


Auto-increment 会 在 新 记录 插入 表 中 时 生成 一 个 唯一 的 数字 。 


AUTO INCREMENT Ff% 


我 们 通常 希望 在 每 次 插入 新 记录 时 ， 自 动 地 创建 主键 字段 的 值 。 
我 们 可 以 在 表 中 创建 一 个 auto-increment 字段 。 


用 于 MySQL 的 语法 
下 列 SQL 语句 把 "Persons" 表 中 的 "P_ld" 列 定义 为 auto-increment 主键 : 


CREATE TABLE Persons 


( 

P_Id int NOT NULL AUTO_INCREMENT, 
LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar(255), 

PRIMARY KEY (P_Id) 

) 


MySQL 使 用 AUTO_INCREMENT 关键 字 来 执行 auto-increment (£4. 
默认 地 ，AUTO_INCREMENT 的 开始 值 是 1， 每 条 新 记录 递增 1。 
要 让 AUTO_INCREMENT 序列 以 其 他 的 值 起 始 ， 请 使 用 下 列 SQL 语法 : 


ALTER TABLE Persons AUTO_INCREMENT=100 


要 在 0 我 们 不 必 为 "P_ld" 列 规定 值 《会 自动 添加 一 个 唯 
Aya): 


INSERT INTO Persons (FirstName, LastName) 
VALUES ('Bill', 'Gates') 


上 面 的 SQL 语句 会 在 "Persons" 表 中 插入 一 条 新 记录 。"P_ld" 会 被 赋予 一 个 唯一 
的 值 。"FirstName" 会 被 设置 为 "Bill"，"LastName" 列 会 被 设置 为 "Gates"。 


用 于 SQL Server 的 语法 
下 列 SQL 语句 把 "Persons" HB "P_Id" 列 定义 为 auto-increment 主键 : 


CREATE TABLE Persons 


( 
P_Id int PRIMARY KEY IDENTITY, 


LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar (255) 

) 


MS SQL 使 用 IDENTITY 关键 字 来 执行 auto-increment £4. 

默认 地 ，IDENTITY 的 开始 值 是 1， 每 条 新 记录 递增 1 

要 规定 "P_ld" 列 以 20 起 始 且 递增 10， 请 把 identity 改 为 IDENTITY(20,10) 

a eee 我 们 不 必 为 "P_ld" 列 规定 值 〈 会 自动 添加 一 个 唯 


INSERT INTO Persons (FirstName, LastName) 
VALUES ('Bill', 'Gates') 


上 面 的 SQL 语句 会 在 "Persons" 表 中 插入 一 条 新 记录 。"P_ld" 会 被 赋予 一 个 唯一 
的 值 。"FirstName" 会 被 设置 为 "Bi"，"LastName" 列 会 被 设置 为 "Gates"。 


用 于 Access 的 语法 
下 列 SQL 语句 把 "Persons" 表 中 的 "P_ld" 列 定义 为 auto-increment 主键 : 


CREATE TABLE Persons 

( 

P_Id int PRIMARY KEY AUTOINCREMENT, 
LastName varchar(255) NOT NULL, 
FirstName varchar(255), 

Address varchar(255), 

City varchar (255) 

) 


MS Access 使 用 AUTOINCREMENT 关键 字 来 执行 auto-increment 任务 。 
默认 地 ，AUTOINCREMENT 的 开始 值 是 1， 每 条 新 记录 递增 


要 规定 "P_ld" 列 以 20 起 始 且 递增 10， 请 把 autoincrement 改 为 
AUTOINCREMENT(20,10) 


要 在 an ee 我 们 不 必 为 "P_ld" 列 规定 值 (会 自动 添加 一 个 唯 
一 的 值 ) : 


INSERT INTO Persons (FirstName, LastName) 
VALUES ('Bill', 'Gates') 


上 面 的 SQL 语句 会 在 "Persons" 表 中 插入 一 条 新 记录 。"P_ld" 会 被 赋予 一 个 唯一 
的 值 。"FirstName" 会 被 设置 为 "Bill"，"LastName" 列 会 被 设置 为 "Gates"。 


用 于 Oracle 的 语法 


在 Oracle 中 ， 代 码 稍 微 复 杀 一 点 。 
您 必须 通过 sequence 对 创建 auto-increment 字段 (该 对 象 生 成 数字 序列 ) 。 
请 使 用 下 面 的 CREATE SEQUENCE 语法 : 


CREATE SEQUENCE seq_person 
MINVALUE 1 

START WITH 1 

INCREMENT BY 1 

CACHE 10 


上 面 的 代码 创建 名 为 seq_person 的 序列 对 象 ， 它 以 1 起 始 且 以 1 递增 。 该 对 象 缓 
存 10 个 值 以 提高 性 能 。CACHE 选项 规定 了 为 了 提高 访问 速度 要 存储 多 少 个 序列 
值 。 


要 在 "Persons" 表 中 插入 新 记录 ， 我 们 必须 使 用 nextval WA GABWAM 
seq_person 序列 中 取 回 下 一 个 值 ) : 


INSERT INTO Persons (P_Id,FirstName,LastName) 
VALUES (seq_person.nextval, 'Lars', 'Monsen' ) 


上 面 的 SQL 语句 会 在 "Persons" 表 中 插入 一 条 新 记录 。"P_ld" 的 赋值 是 来 自 
seq_person 序 序列 的 下 一 TRF. "FirstName" 会 被 设置 为 "Bil"，"LastName" 列 会 
被 设置 为 "Gates"。 


SQL VIEW (视图 ) 


视图 是 可 视 化 的 表 。 
本 章 讲 解 如 何 创建 、 更 新 和 删除 视图 。 


SQL CREATE VIEW 语句 


什么 是 视图 ? 

在 SQL 中 ， 视 图 是 基于 SQL 语句 的 结果 集 的 可 视 化 的 表 。 

视图 包含 行 和 列 ， 就 像 一 个 真实 的 表 。 视 图 中 的 字段 就 是 来 自 一 个 或 多 个 数据 库 中 
的 真实 的 表 中 的 字段 。 我 们 可 以 向 视图 添加 SQL WA. WHERE 以 及 JOIN 语句 ， 
我 们 也 可 以 提交 数据 ， 就 像 这 些 来 自 于 某 个 单一 的 表 。 

注释 : 数据 库 的 设计 和 结构 不 会 受到 视图 中 的 函数 、where 或 join 语句 的 影响 。 


SQL CREATE VIEW 语法 


CREATE VIEW view_name AS 
SELECT column_name(s) 
FROM table_name 

WHERE condition 


注释 : 视图 总 是 显示 最 近 的 数据 。 每 当 用 户 查 询 视 图 时 ， 数 据 库 引 擎 通过 使 用 SQL 
语句 来 重建 数据 。 


SQL CREATE VIEW 实例 


可 以 从 某 个 查询 内 部 、 某 个 存储 过 程 内 部 ， 或 者 从 另 一 个 视图 内 部 来 使 用 视图 。 通 
过 向 视图 添加 西数 、join 等 等 ， 我 们 可 以 向 用 户 精确 地 提交 我 们 希望 提交 的 数据 。 


样本 数据 库 Northwind 拥有 一 些 被 默认 安装 的 视图 。 视 图 "Current Product List" 会 
从 Products 表 列 出 所 有 正在 使 用 的 产品 。 这 个 视图 使 用 下 列 SQL 创建 : 


CREATE VIEW [Current Product List] AS 
SELECT ProductID,ProductName 

FROM Products 

WHERE Discontinued=No 


我 们 可 以 查询 上 面 这 个 视图 : 


SELECT * FROM [Current Product List] 


Northwind 样本 数据 库 的 另 一 个 视图 会 选取 Products 表 中 所 有 单位 价格 高 于 平均 单 
位 价格 的 产品 : 


CREATE VIEW [Products Above Average Price] AS 

SELECT ProductName, UnitPrice 

FROM Products 

WHERE UnitPrice>(SELECT AVG(UnitPrice) FROM Products) 


我 们 可 以 像 这 样 查询 上 面 这 个 视图 : 


SELECT * FROM [Products Above Average Price] 


另 一 个 来 自 Northwind 数据 库 的 视图 实例 会 计算 在 1997 年 每 个 种 类 的 销售 总 数 。 
请 注意 ， 这 个 视图 会 从 另 一 个 名 为 "Product Sales for 1997" 的 视图 那里 选取 数 
据 : 


CREATE VIEW [Category Sales For 1997] AS 

SELECT DISTINCT CategoryName, Sum(ProductSales) AS CategorySales 
FROM [Product Sales for 1997] 

GROUP BY CategoryName 


我 们 可 以 像 这 样 查询 上 面 这 个 视图 : 


SELECT * FROM [Category Sales For 1997] 


我 们 也 可 以 向 查询 添加 条 件 。 现 在 ， 我 们 仅仅 需要 查看 "Beverages" 类 的 全 部 销 


== 


SELECT * FROM [Category Sales For 1997] 
WHERE CategoryName='Beverages' 


SQL 更 新 视图 
您 可 以 使 用 下 面 的 语法 来 更 新 视图 : 


SQL CREATE OR REPLACE VIEW Syntax 
CREATE OR REPLACE VIEW view_name AS 
SELECT column_name(s) 

FROM table_name 

WHERE condition 


现在 ， 我 们 希望 向 "Current Product List" 视图 添加 "Category" 列 。 我 们 将 通过 下 
列 SQL 更 新 视图 : 


CREATE VIEW [Current Product List] AS 
SELECT ProductID, ProductName, Category 
FROM Products 

WHERE Discontinued=No 


SQL 撤销 视 
您 可 以 通过 DROP VIEW 命令 来 删除 视图 。 


SQL DROP VIEW Syntax 
DROP VIEW view_name 


SQL 


SQL Date 函数 


SQL 日 期 


当 我 们 处 理 日 期 时 ， 最 难 的 任务 恐怕 是 确保 所 插入 的 日 期 的 格式 ， 与 数据 库 中 日 期 
列 的 格式 相 匹 配 。 


只 要 数据 包含 的 只 是 日 期 部 分 ， 运 行 查询 就 不 会 出 问题 。 但 是 ， 如 果 涉 及 时 间 ， 情 
况 就 有 点 复杂 了 。 


在 讨论 日 期 查询 的 复杂 性 之 前 ， 我 们 先 来 看 看 最 重要 的 内 建 日 期 处理 芳 数 。 


MySQL Date 函数 
下 面 的 表格 列 出 了 MySQL Pre SAS AR : 


NOW() 返回 当前 的 日 期 和 时 间 

CURDATE() 返回 当前 的 日 期 

CURTIME() 返回 当前 的 时 间 

DATE() 提取 日 期 或 日 期 /时 间 表 达 式 的 日 期 部 分 
EXTRACT() 返回 日 期 /时 间 按 的 单独 部 分 
DATE_ADD() 给 日 期 添加 指定 的 时 间 间 隔 
DATE_SUB() 从 日 期 减 去 指定 的 时 间 间 隅 
DATEDIFF() 返回 两 个 日 期 之 间 的 天 数 
DATE_FORMAT() 用 不 同 的 格式 显示 日 期 /时 间 


SQL Server Date 函数 
下 面 的 表格 列 出 了 SQL Server 中 最 重要 的 内 建 日 期 范 数 : 


GETDATE() 返回 当前 日 期 和 时 间 
DATEPART() 返回 日 期 /时 间 的 单独 部 分 
DATEADD() 在 日 期 中 添加 或 减 去 指定 的 时 间 间 陋 
DATEDIFF() 返回 两 个 日 期 之 间 的 时 间 
CONVERT() 用 不 同 的 格式 显示 日 期 /时 间 

SQL Date 数据 类 型 


MySQL 使 用 下 列 数据 类 型 在 数据 库 中 存储 日 期 或 日 期 /时 间 值 : 


DATE - 格式 YYYY-MM-DD 

DATETIME - 格式 : YYYY-MM-DD HH:MM:SS 
TIMESTAMP - 格式 : YYYY-MM-DD HH:MM:SS 
YEAR - 格式 YYYY 或 YY 


SQL Server 使 用 下 列 数据 类 型 在 数据 库 中 存储 日 期 或 日 期 /时 间 值 : 


DATE - 格式 YYYY-MM-DD 

DATETIME - 格式 : YYYY-MM-DD HH:MM:SS 
SMALLDATETIME - 格式 : YYYY-MM-DD HH:MM:SS 
TIMESTAMP - 格式 : 唯一 的 数字 


SQL 日 期 处 理 


如 果 不 涉及 时 间 部 分 ， 那 么 我 们 可 以 轻松 地 上 比较 两 个 日 期 ! 
假设 我 们 有 下 面 这 个 "Orders" R : 


Orderld ProductName OrderDate 
1 computer 2008-12-26 
2 printer 2008-12-26 
3 electrograph 2008-11-12 
4 telephone 2008-10-19 


现在 ， 我 们 希望 从 上 表 中 选取 OrderDate 为 "2008-12-26" 的 记录 。 
我 们 使 用 如 下 SELECT 语句 : 


SELECT * FROM Orders WHERE OrderDate='2008-12-26' 


Orderld ProductName OrderDate 
1 computer 2008-12-26 
3 electrograph 2008-12-26 


现在 假设 "Orders" 类 似 这 桩 (请 注意 "OrderDate" 列 中 的 时 间 部 分 ) 


Orderld ProductName OrderDate 
1 computer 2008-12-26 16:23:55 
2 printer 2008-12-26 10:45:26 
3 electrograph 2008-11-12 14:12:08 
4 telephone 2008-10-19 12:56:10 


如 果 我 们 使 用 上 面 的 SELECT 语句 : 


SELECT * FROM Orders WHERE OrderDate='2008-12-26' 


那么 我 们 得 不 到 结果 。 这 是 由 于 该 查询 不 含有 时 间 部 分 的 日 期 。 
提示 : 如 果 您 希望 使 查询 简单 且 更 易 维护 ， 那 么 请 不 要 在 日 期 中 使 用 时 间 部 分 ! 


SQL NULL 值 


NULL 值 是 遗漏 的 未 知 数据 。 
默认 地 ， 表 的 列 可 以 存放 NULL 值 。 
本 章 讲 解 IS NULL 和 IS NOT NULL 操作 符 。 


SQL NULL 值 


如 果 表 中 的 某 个 列 是 可 选 的 ， 那 么 我 们 可 以 在 不 向 该 列 添加 值 的 情况 下 插入 新 记录 
或 更 新 已 有 的 记录 。 这 意味 着 该 字段 将 以 NULL 值 保 存 。 

NULL 值 的 处 理 方式 与 其 他 值 不 同 。 

NULL 用 作 未 知 的 或 不 适用 的 值 的 占 位 符 。 

注释 : 无 法 比较 NULL 和 0 ; 它们 是 不 等 价 的 。 


SQL 的 NULL 值 处 理 


请 看 下 面 的 "Persons" 表 : 


Id LastName FirstName Address City 
1 Adams John London 
2 Bush George Fifth Avenue New York 
3 Carter Thomas Beijing 


{40 "Persons" 表 中 的 "Address" 列 是 可 选 的 。 这 意味 着 如 果 在 "Address" 列 插入 
一 条 不 带 值 的 记录 ，"Address'" 列 会 使 用 NULL HIRE. 


那么 我 们 如 何 测试 NULL 值 呢 ? 
无 法 使 用 比较 运算 符 来 测试 NULL 值 ， 比 如 =, <, 或 者 <>, 
我 们 必须 使 用 IS NULL 和 IS NOT NULL 操作 符 。 


SQL IS NULL 


我 们 如 何 仅 仅 选 取 在 "Address" 列 中 带 有 NULL 值 的 记录 呢 ? 
我 们 必须 使 用 IS NULL 操作 符 : 


SELECT LastName, FirstName, Address FROM Persons 
WHERE Address IS NULL 


结果 集 
LastName FirstName Address 
Adams John 
Carter Thomas 


提示 : 请 始终 使 用 IS NULL 来 查找 NULL 值 。 


SQL IS NOT NULL 


我 们 如 何 选取 在 "Address" 列 中 不 带 有 NULL 值 的 记录 呢 ? 
我 们 必须 使 用 IS NOT NULL 操作 符 : 


SELECT LastName,FirstName,Address FROM Persons 
WHERE Address IS NOT NULL 


LastName FirstName Address 


Bush George Fifth Avenue 


在 下 一 节 中 ， 我 们 了 解 I SNULL()、NVL()、IFNULL() 和 COALESCE() 函数 。 


SQL NULL WH 

SQL ISNULL(), NVL(), IFNULL() 和 COALESCE() 
函数 

请 看 下 面 的 "Products" X : 


P_ld ProductName UnitPrice UnitsInStock UnitsOnOrder 


1 computer 699 25 15 
2 printer 365 36 
3 telephone 280 159 57 


假如 "UnitsOnOrder" 是 可 选 的 ， 而 且 可 以 包含 NULL 值 。 
我 们 使 用 如 下 SELECT 语句 : 


SELECT ProductName,UnitPrice*(UnitsInStock+Unitsonorder ) 
FROM Products 


在 上 面 的 例子 中 ， 如 果 有 "UnitsOnOrder" 值 是 NULL, WARE NULL. 
微软 的 ISNULL() 画 数 用 于 规定 如 何 义理 NULL 值 。 

NVL(), IFNULL() 和 COALESCE() 函数 也 可 以 达到 相同 的 结果 。 

在 这 里 ， 我 们 希望 NULL 值 为 0。 

下 面 ， 如 果 "UnitsOnOrder" 是 NULL， 则 不 利于 计算 ， 因 此 如 果 值 是 NULL 则 
ISNULL() 返回 0。 

SQL Server / MS Access 


SELECT ProductName, UnitPrice*(UnitsInStock+ISNULL(UnitsOnOrder, 0) ) 
FROM Products 
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Oracle 


Oracle 没有 ISNULL() 函数 。 不 过 ， 我 们 可 以 使 用 NVL() 函数 达到 相同 的 结果 : 


SELECT ProductName,UnitPrice*(UnitsInStock+NVL(Unitsonorder,0)) 
FROM Products 


MySQL 


MySQL 也 拥有 类 似 ISNULL() 的 函数 。 不 过 它 的 工作 方式 与 微软 的 ISNULL() RX 
有 点 不 同 。 


在 MySQL 中 ， 我 们 可 以 使 用 IFNULL() EBX, BRR ES: 


SELECT ProductName,UnitPrice*(UnitsInStock+IFNULL(UnitsOnorder,0)) 
FROM Products 
二 = AA 
或 者 我 们 可 以 使 用 COALESCE() 函数 ， 就 像 这 样 : 


SELECT ProductName,UnitPrice*(UnitsInStock+COALESCE(UnitsOnOrder,0 
FROM Products 


A e 





SQL 数据 类 型 


Microsoft Access、MySQL 以 及 SQL Server 所 使 用 的 数据 类 型 和 范围 。 


Microsoft Access 数据 类 型 


数据 类 型 
Text 


Memo 


Byte 


Integer 


Long 


Single 


Double 


Currency 


AutoNumber 


Date/Time 


Yes/No 


Ole Object 


Hyperlink 


Lookup 
Wizard 


描述 
用 于 文本 或 文本 与 数字 的 组 合 。 最 多 255 个 字符 。 
Memo 用 于 更 大 数量 的 文本 。 最 多 存储 65,536 个 字 


符 。 注释 : 无 法 对 memo 字段 进行 排序 。 不 过 它们 是 
可 搜索 的 。 


人 允许 0 到 255 的 数字 。 


人 允许 介 于 -32,768 到 32,767 之 间 的 数字 。 

人 允许 介 于 -2,147,483,648 与 2,147,483,647 之 间 的 全 部 
数字 

单 精度 浮 点 。 义 理 大 多 数 小 数 。 


双 精 度 浮 点 。 处 理 大 多 数 小 数 。 


FAT i fh. Sete 15 位 的 元 ， 外 加 4 位 小 数 。 提示 : 您 


可 以 选择 使 用 哪个 国家 的 货币 。 

AutoNumber 字段 自动 为 每 条 记录 分 配 数 字 ， 通 常 从 1 
开始 。 

用 于 日 期 和 时 间 


逻辑 字段 ， 可 以 显示 为 Yes/No、True/False 或 
On/Off。 在 代码 中 ， 使 用 常量 True 和 False (等 价 于 
1 和 0) 注释 : Yes/No 字段 中 不 允许 Null 值 


可 以 存储 图 片 、 音 频 、 视 频 或 其 他 BLOBs (Binary 
Large OBjects) 


包含 指向 其 他 文件 的 链接 ， 包 括 网 页 。 
允许 你 创建 一 个 可 从 下 列 列表 中 进行 选择 的 选项 列表 。 


存储 


tt} # # 


4} 


tO De HO a PO ay ar any oS 
hi} hi} hi} hi} 


AF 一 人 
at 
Cv 


MySQL 数据 类 型 
在 MySQL 中 ， 有 三 种 主要 的 类 型 : 文本 、 数 字 和 日 期 /时 间 类 型 。 


Text 类 型 : 
数据 类 型 


CHAR(size) 


VARCHAR(size) 


TINYTEXT 
TEXT 


BLOB 
MEDIUMTEXT 
MEDIUMBLOB 
LONGTEXT 


LONGBLOB 


ENUM(x,y,z,etc. ) 


SET 


Number 类 型 : 


Hah 


保存 固定 长 度 的 字符 串 〈 可 包含 字母 、 数 字 以 及 特殊 字 
F) 。 在 括号 中 指定 字符 串 的 长 度 。 最 多 255 个 字符 。 


保存 可 变 长 度 的 字符 串 〈 可 包含 字母 、 数 字 以 及 特殊 字 
符 ) 。 在 括号 中 指定 字符 串 的 最 大 长 度 。 最 多 255 个 字 
符 。 注释 : 如 果 值 的 长 度 大 于 255， 则 被 转换 为 TEXT 类 
型 。 


存放 最 大 长 度 为 255 个 字符 的 字符 串 。 
存放 最 大 长 度 为 65,535 个 字符 的 字符 串 。 


用 于 BLOBs (Binary Large OBjects)。 存 放 最 多 65,535 字 
节 的 数据 。 


存放 最 大 长 度 为 16,777,215 个 字符 的 字符 串 。 


用 于 BLOBs (Binary Large OBjects)。 存 放 最 多 
16,777,215 字 节 的 数据 。 


存放 最 大 长 度 为 4,294,967,295 个 字符 的 字符 串 。 


用 于 BLOBs (Binary Large OBjects)。 存 放 最 多 
4,294,967,295 字 节 的 数据 。 


允许 你 输入 可 能 值 的 列表 。 可 以 在 ENUM 列表 中 列 出 最 大 
65535 个 值 。 如 果 列 表 中 不 存在 插入 的 值 ， 则 插入 空 值 。 
注释 : 这 些 值 是 按照 你 输入 的 顺序 存储 的 。 可 以 按照 此 格 
式 输 入 可 能 的 值 : ENUM('X','Y','Z') 


与 ENUM 类 似 ，SET 最 多 只 能 包含 64 个 列表 项 ， 不 过 
SET 可 存储 一 个 以 上 的 值 。 


数据 类 型 


TINYINT(size) 


SMALLINT(size) 


MEDIUMINT(size) 


INT(size) 


BIGINT(size) 


FLOAT(size,d) 


DOUBLE (size,d) 


DECIMAL(size,d) 


-128 到 127 常规 。0 到 255 无 符号 *。 在 括号 中 规定 最 大 
位 数 。 

-32768 到 32767 常规 。0 到 65535 无 符号 *。 在 括号 中 
规定 最 大 位 数 。 

-8388608 到 8388607 普通 。0 to 16777215 无 符号 *。 在 
括号 中 规定 最 大 位 数 。 


-2147483648 到 2147483647 常规 。0 到 4294967295 无 
符号 *。 在 括号 中 规定 最 大 位 数 。 


-9223372036854775808 到 9223372036854775807 常 
规 。0 到 18446744073709551615 无 符号 *。 在 括号 中 规 
定 最 大 位 数 。 


带 有 浮动 小 数 点 的 小 数字 。 在 括号 中 规定 最 大 位 数 。 在 d 
参数 中 规定 小 数 点 右 侧 的 最 大 位 数 。 


带 有 浮动 小 数 点 的 大 数字 。 在 括号 中 规定 最 大 位 数 。 在 d 
参数 中 规定 小 数 点 右 侧 的 最 大 位 数 。 


作为 字符 串 存储 的 DOUBLE 类 型 ， 人 允许 固定 的 小 数 点 。 


o 这 些 整数 类 型 拥有 额外 的 选项 UNSIGNED。 通 常 ， 整 数 可 以 是 负数 或 正 数 。 
如 果 添 加 UNSIGNED 属性 ， 那 么 范围 将 从 0 开始 ， 而 不 是 某 个 负数 。 


Date 类 型 : 
数据 类 型 


DATE() 


描述 


日 期 。 格 式 : YYYY-MM-DD 注释 : 支持 的 范围 是 从 '1000- 
01-01' 到 '9999-12-31' 


* 日 期 和 时 间 的 组 合 。 格 式 : YYYY-MM-DD HH:MM:SS 注 
DATETIME() 释 : 支持 的 范围 是 从 '1000-01-01 00:00:00' 到 '9999-12-31 
23:59:59' 


*at Ho TIMESTAMP 和 值 使 用 Unix 纪元 (1970-01-01 


TIMESTAMP() 


00:00:00' UTC) 至 今 的 描述 来 存储 。 格 式 : YYYY-MM-DD 
HH:MM:SS 注释 : 支持 的 范围 是 从 '1970-01-01 00:00:01' 


UTC 到 '2038-01-09 03:14:07' UTC 


TIME() 


时 间 。 格 式 : HH:MM:SS 注释 : 支持 的 范围 是 从 '-838:59:59' 
到 '838:59:59' 


2 位 或 4 位 格式 的 年 。 注释 : 4 位 格式 所 允许 的 值 : 1901 到 
YEAR() 2155. 2 位 格式 所 允许 的 值 : 70 B69, HARM 1970 到 
2069。 


e 即便 DATETIME 和 TIMESTAMP 返回 相同 的 格式 ， 它 们 的 工作 方式 很 不 同 。 
在 INSERT 或 UPDATE 查询 中 ，TIMESTAMP 自动 把 自身 设置 为 当前 的 日 期 
和 时 间 。TIMESTAMP 也 接受 不 同 的 格式 ， 比 如 YYYYMMDDHHMMSS、 
YYMMDDHHMMSS、YYYYMMDD 或 YYMMDD. 


SQL Server 数据 类 型 


Character 字符 串 : 
数据 类 型 描述 
char(n) 固定 长 度 的 字符 串 。 最 多 8,000 个 字符 。 
varchar(n) 可 变 长 度 的 字符 串 。 最 多 8,000 个 字符 。 
varchar(max) ”可 变 长 度 的 字符 串 。 最 多 1,073,741,824 个 字符 。 
text 可 变 长 度 的 字符 串 。 最 多 2GB 字符 数据 。 
Unicode 字符 串 : 
数据 类 型 描述 
nchar(n) EE KEKI Unicode 数据 。 最 多 4,000 个 字符 。 
nvarchar(n) FB] = KE Unicode 数据 。 最 多 4,000 个 字符 。 
nvarchar(max) a 长 度 的 Unicode 数据 。 最 多 536,870,912 个 字 
ntext 可 变 长 度 的 Unicode 数据 。 最 多 2GB 字符 数据 。 


Binary 类 型 : 


数据 类 型 
bit 
binary(n) 
varbinary(n) 
varbinary(max) 


image 


人 允许 0. 1 BK NULL 

固定 长 度 的 二 进 制 数 据 。 
可 变 长 度 的 二 进 制 数据 。 
可 变 长 度 的 二 进 制 数据 。 


描述 
最 多 8,000 字 节 。 
最 多 8,000 字 节 。 
最 多 2GB 字 节 。 
最 多 2GB。 


可 变 长 度 的 二 进 制 数据 。 


Number 类 型 : 


存储 


存储 


数据 类型 


tinyint 


smallint 


int 


bigint 


decimal(p,s) 


numeric(p,s) 


smallmoney 


money 


float(n) 


real 


Date 类 型 : 


描 


学 


允许 从 0 到 255 的 所 有 数字 。 


允许 从 -32,768 到 32,767 的 所 有 数字 。 


允许 从 -2,147,483,648 到 2,147,483,647 的 所 有 数字 。 


允许 介 于 -9,223,372,036,854,775,808 和 
9,223,372,036,854,775,807 之 间 的 所 有 数字 。 


固定 精度 和 上 比例 的 数字 。 人 允许 从 -10^38 +1 到 10438 -1 
之 间 的 数字 。 p 参数 指示 可 以 存储 的 最 大 位 数 (小 数 点 左 
侧 和 右 侧 ) 。p 必须 是 1 到 38 之 间 的 值 。 默 认 是 18。 s 
参数 指示 小 数 点 右 侧 存储 的 最 大 位 数 。s 必须 是 0 到 p 之 
间 的 值 。 默 认 是 0。 


固定 精度 和 上 比例 的 数字 。 人 允许 从 -10^38 +1 到 10^38 -1 
之 间 的 数字 。 p 参数 指示 可 以 存储 的 最 大 位 数 (小 数 点 左 
侧 和 右 侧 ) 。p 必须 是 1 到 38 之 间 的 值 。 默 认 是 18。 s 
参数 指示 小 数 点 右 侧 存储 的 最 大 位 数 。s 必须 是 0 到 p 之 
间 的 值 。 默 认 是 0。 


NF -214,748.3648 和 214,748.3647 之 间 的 货币 数据 。 


介 于 -922,337,203,685,477.5808 和 
922,337,203,685,477.5807 之 间 的 货币 数据 。 


> 


人 -1.79E + 308 到 1.79E + 308 的 浮动 精度 数字 数据 。 
参数 n 指示 该 字段 保存 4 字 节 还 是 8 FH. float(24) 保 
存 4 字 节 ， 而 float(53) 保存 8 字 节 。n 的 默认 值 是 53。 


Y 


M -3.40E + 38 到 3.40E + 38 的 浮动 精度 数字 数据 。 


HRL I aR RA 


aS 


aS 9 


eR aOR a HO aE 


数据 类 型 描述 存储 


datetime 从 1753 年 1 月 1 日 到 9999 年 12 月 31 日 ， 精 度 8 


为 3.33 毫秒 。 bytes 
| M 1753 £1818 3/9999 12 A 31A, BE 6-8 
seredan 为 100 纳 秒 。 bytes 
smalidatetime ， 从 1900 年 1 月 1 日 到 2079 年 6 月 6 日 精度 为 1 4 
分 钟 。 bytes 
仅 存 储 日 期 。 从 0001 年 1 月 1 日 到 9999 年 12 月 3 
ate 
Ole bytes 
time 仅 存储 时 间 。 精 度 为 100 纳 秒 。 es 
8-10 


datetimeoffset ”与 datetime2 相同 ， 外 加 时 区 偏 移 。 


存储 唯一 的 数字 ， 每 当 创 建 或 修改 某 行 时 ， 该 数字 会 
timestamp 更 新 。timestamp 基于 内 部 时 钟 ， 不 对 应 真实 时 间 。 
每 个 表 只 能 有 一 个 timestamp 变量 。 


其 他 数据 类 型 : 
数据 类 型 par 
valni 存储 最 多 8,000 字 节 不 同 数据 类 型 的 数据 ， 除 了 text, ntext 


以 及 timestamp。 
uniqueidentifier ， 存 储 全 局 标识 符 (GUID). 


xml 存储 XML 格式 化 数据 。 最 多 2GB。 
cursor 存储 对 用 于 数据 库 操 作 的 指针 的 引用 。 


table 存储 结果 集 ， 供 稳 后 人 处理。 


SQL fk 44s -RDBMS 


现代 的 SQL 服务 器 构建 在 RDBMS 之 上 。 


DBMS - 数据 库 管 理 系 统 (Database Management 
System) 


数据 库 管理 系统 是 一 种 可 以 访问 数据 库 中 数据 的 计算 机 程序 。 
DBMS 使 我 们 有 能 力 在 数据 库 中 提取 、 修 改 或 者 存 贮 信息 。 
不 同 的 DBMS 提供 不 同 的 函数 供 查询 、 提 交 以 及 修改 数据 。 


RDBMS - 关系 数据 库 管理 系统 (Relational Database 
Management System) 


关系 数据 库 管理 系统 (RDBMS) 也 是 一 种 数据 库 管理 系统 ， 其 数据 库 是 根据 数据 间 
的 关系 来 组 织 和 访问 数据 的 。 


20 世纪 70 年 代 初 ，IBM 公司 发 明了 RDBMS。 


RDBMS 是 SQL 的 基础 ， 也 是 所 有 现代 数据 库 系 统 诸 如 Oracle、SQL Server、 
IBM DB2, Sybase, MySQL LA Microsoft Access 的 基础 。 


SQL IX 

SQL 拥有 很 多 可 用 于 计数 和 计算 的 内 建 酚 数 。 
函数 的 语法 

A SQL 函数 的 语法 是 : 


SELECT function( 列 ) FROM 表 


函数 的 类 型 
在 SQL 中 ， 基 本 的 函数 关 型 和 种 关 有 若干 种 。 丽 数 的 基本 类 型 是 : 


e Aggregate HX 
e Scalar HA 


合计 函数 (Aggregate functions) 


Aggregate 函数 的 操作 面向 一 系列 的 值 ， 并 返回 一 个 单一 的 值 。 


注释 : 如 果 在 SELECT 语句 的 项 目 列表 中 的 众多 其 它 表 达 式 中 使 用 SELECT 4 
句 ， 则 这 个 SELECT 必须 使 用 GROUP BY 语句 ! 


"Persons" table (在 大 部 分 的 例子 中 使 用 过 ) 


Name Age 
Adams, John 38 
Bush, George 33 
Carter, Thomas 28 


MS Access 中 的 合计 函数 


AVG(column) 返回 某 列 的 平均 值 
COUNT(column) 返回 某 列 的 行 数 〈 不 包括 NULL 值 ) 
COUNT(*) 返回 被 选 行 数 

FIRST(column) 返回 在 指定 的 域 中 第 一 个 记录 的 值 
LAST(column) 返回 在 指定 的 域 中 最 后 一 个 记录 的 值 
MAX(column) 返回 某 列 的 最 高 值 

MIN(column) 返回 某 列 的 最 低 值 
STDEV(column) 

STDEVP(column) 

SUM(column) 返回 某 列 的 总 和 

VAR(column) 

VARP(column) 


在 SQL Server 中 的 合计 函数 


函数 描述 
AVG(column) 返回 某 列 的 平均 值 
BINARY_CHECKSUM 
CHECKSUM 
CHECKSUM AGG 
COUNT(column) 返回 某 列 的 行 数 〈 不 包括 NULL 值 ) 
COUNT(*) 返回 被 选 行 数 
CU ”” “返回 相 异 结果 的 数目 


返回 在 指定 的 域 中 第 一 个 记录 的 值 (SQLServer2000 
不 文 持 ) 


返回 在 指定 的 域 中 最 后 一 个 记录 的 值 


FIRST(column) 


i ila (SQLServer2000 不 支持 ) 
MAX(column) 返回 某 列 的 最 高 值 
MIN(column) 返回 某 列 的 最 低 值 
STDEV(column) 

STDEVP(column) 

SUM(column) 返回 某 列 的 总 和 
VAR(column) 

VARP(column) 


Scalar HX Scalar 函数 的 操作 面向 某 个 单一 的 值 ， 并 
返回 基于 输入 值 的 一 个 单一 的 值 。 ##H MS Access 中 
的 Scalar HA 


HŽ 
UCASE(c) 
LCASE(c) 
MID(c,start[,end]) 
LEN(c) 
INSTR(c,char) 
LEFT(c,number_of_char) 
RIGHT(c,number_of_char) 
ROUND(c,decimals) 
MOD(x,y) 
NOW() 
FORMAT (c,format) 
DATEDIFF(d,date1 ,date2) 


措 ; 
将 某 个 域 转换 为 大 写 
将 某 个 域 转换 为 小 写 

从 某 个 文本 域 提取 字符 
返回 某 个 文本 域 的 长 度 
返回 在 某 个 文本 域 中 指定 字符 的 数值 位 置 
返回 某 个 被 请 求 的 文本 域 的 左 侧 部 分 

返回 某 个 被 请 求 的 文本 域 的 右 侧 部 分 

对 某 个 数值 域 进行 指定 小 数位 数 的 四 舍 五 入 
返回 除法 操作 的 余数 

返回 当前 的 系统 日 期 

改变 某 个 域 的 显示 方式 

用 于 执行 日 期 计算 


Er 


SQL AVG WHR 


定义 和 用 法 


AVG WOR ERA JIII FHA. NULL 值 不 包括 在 计算 中 。 


SQL AVG() 语法 


SELECT AVG(column_name) FROM table _ name 


SQL AVG() 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_Id OrderDate 


1 2008/12/29 
2008/11/23 
2008/10/05 
2008/09/28 
2008/08/06 
2008/07/21 


O oO FPF W N 


例子 1 


OrderPrice 
1000 
1600 
700 
300 
2000 
100 


现在 ， 我 们 希望 计算 "OrderPrice" 字段 的 平均 值 。 


我 们 使 用 如 下 SQL 语句 : 


Customer 
Bush 
Carter 
Bush 
Bush 
Adams 


Carter 


SELECT AVG(OrderPrice) AS OrderAverage FROM Orders 


结果 集 类 似 这 样 : 


950 


OrderAverage 


例子 2 
现在 ， 我 们 希望 找到 OrderPrice 值 高 于 OrderPrice 平均 值 的 客户 。 


我 们 使 用 如 下 SQL 语句 : 


SELECT Customer FROM Orders 
WHERE OrderPrice>(SELECT AVG(OrderPrice) FROM Orders) 


结果 集 类 似 这 样 : 


Customer 
Bush 
Carter 


Adams 


SQL COUNT() 函数 


COUNT() 画 数 返回 匹配 指定 条 件 的 行 数 。 
SQL COUNT() 语法 


SQL COUNT(column_name) 语法 


COUNT(column_name) AGREE EFUB aes (NULL Fit A) 


SELECT COUNT(column_name) FROM table_name 


SQL COUNT(*) 语法 
COUNT(*) HUREK AJE ER : 


SELECT COUNT(*) FROM table_name 


SQL COUNT(DISTINCT column_name) 语法 
COUNT(DISTINCT column_name) 函数 返回 指定 列 的 不 同 值 的 数目 : 


SELECT COUNT(DISTINCT column_name) FROM table_name 


注释 : COUNT(DISTINCT) 适用 于 ORACLE 和 Microsoft SQL Server， 但 是 无 法 
用 于 Microsoft Access。 


SQL COUNT(column_name) 实例 


我 们 拥有 下 列 "Orders" K : 


OrderDate OrderPrice Customer 
1 2008/12/29 1000 Bush 
2 2008/11/23 1600 Carter 
3 2008/10/05 700 Bush 
4 2008/09/28 300 Bush 
5 2008/08/06 2000 Adams 
6 2008/07/21 100 Carter 


现在 ， 我 们 希望 计算 客户 "Carter" 的 订单 数 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT COUNT(Customer) AS CustomerNilsen FROM Orders 
WHERE Customer='Carter' 


以 上 SQL 语句 的 结果 是 2， 因 为 客户 Carter 共有 2 个 订单 : 


CustomerNilsen 


2 
SQL COUNT(*) 实例 
如 果 我 们 省 略 WHERE 子 句 ， 比 如 这 样 : 


SELECT COUNT(*) AS NumberOfOrders FROM Orders 


结果 集 类 似 这 样 : 
NumberOfOrders 
6 


这 是 表 中 的 总 行 数 。 
SQL COUNT(DISTINCT column_name) 实例 


现在 ， 我 们 希望 计算 "Orders" 表 中 不 同 客户 的 数目 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT COUNT(DISTINCT Customer) AS NumberOfCustomers FROM Orders 


结果 集 类 似 这 样 : 


NumberOfCustomers 
3 


这 是 "Orders" 表 中 不 同 客户 (Bush, Carter 和 Adams) 的 数目 。 


eA 


SQL FIRST() 函数 


A 


FIRST() 函数 


FIRST() 函数 返回 指定 的 字段 中 第 一 个 记录 的 值 。 
提示 : 可 使 用 ORDER BY 语句 对 记录 进行 排序 。 


SQL FIRST() 语法 


SELECT FIRST(column_name) FROM table_name 


SQL FIRST() 实例 
我 们 拥有 下 面 这 个 "Orders" X : 


O_Id OrderDate 


1 2008/12/29 
2008/11/23 
2008/10/05 
2008/09/28 
2008/08/06 
2008/07/21 


O ao A W N 


OrderPrice 
1000 
1600 
700 
300 
2000 
100 


现在 ， 我 们 希望 查找 "OrderPrice" 列 的 第 一 个 值 。 


我 们 使 用 如 下 SQL 语句 : 


SELECT FIRST(OrderPrice) AS FirstOrderPrice FROM 


结果 集 类 似 这 样 : 


1000 


FirstOrderPrice 


Customer 
Bush 
Carter 
Bush 
Bush 
Adams 


Carter 


Orders 


eA 


SQL LAST() 函数 


A 


LAST() 函数 


LAST() 函数 返回 指定 的 字段 中 最 后 一 个 记录 的 值 。 
提示 : 可 使 用 ORDER BY 语句 对 记录 进行 排序 。 


SQL LAST() 语法 


SELECT LAST(column_name) FROM table name 


SQL LAST() 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_Id OrderDate OrderPrice Customer 
1 2008/12/29 1000 Bush 
2 2008/11/23 1600 Carter 
3 2008/10/05 700 Bush 
4 2008/09/28 300 Bush 
5 2008/08/06 2000 Adams 
6 2008/07/21 100 Carter 


现在 ， 我 们 希望 查找 "OrderPrice" 列 的 最 后 一 个 值 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT LAST(OrderPrice) AS LastOrderPrice FROM Orders 
结果 集 类 似 这 样 : 


LastOrderPrice 


100 


I 


SQL MAX() HŽ 


A 


MAX() 函数 
MAX 函数 返回 一 列 中 的 最 大 值 。NULL 值 不 包括 在 计算 中 。 
SQL MAX() 语法 

SELECT MAX(column_name) FROM table_name 


注释 : MIN 和 MAX 也 可 用 于 文本 列 ， 以 获得 按 字 母 顺 序 排 列 的 最 高 或 最 低 值 。 


SQL MAX() 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_Id OrderDate OrderPrice Customer 
1 2008/12/29 1000 Bush 
2 2008/11/23 1600 Carter 
3 2008/10/05 700 Bush 
4 2008/09/28 300 Bush 
5 2008/08/06 2000 Adams 
6 2008/07/21 100 Carter 


现在 ， 我 们 希望 查找 "OrderPrice" 列 的 最 大 值 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT MAX(OrderPrice) AS LargestOrderPrice FROM Orders 
结果 集 类 似 这 样 : 


LargestOrderPrice 
2000 


eA 


SQL MIN() 西数 


A 


MIN() 2X 
MIN KARE — FU Pee). NULL 值 不 包括 在 计算 中 。 
SQL MIN() 语法 

SELECT MIN(column_name) FROM table_name 


注释 : MIN 和 MAX 也 可 用 于 文本 列 ， 以 获得 按 字 母 顺 序 排 列 的 最 高 或 最 低 值 。 


SQL MIN() 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_Id OrderDate OrderPrice Customer 
1 2008/12/29 1000 Bush 
2 2008/11/23 1600 Carter 
3 2008/10/05 700 Bush 
4 2008/09/28 300 Bush 
5 2008/08/06 2000 Adams 
6 2008/07/21 100 Carter 


现在 ， 我 们 希望 查找 "OrderPrice" 列 的 最 小 值 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT MIN(OrderPrice) AS SmallestOrderPrice FROM Orders 
结果 集 类 似 这 样 : 


SmallestOrderPrice 


100 


A 


SQL SUM() 函数 

SUM() 2X 

SUM HAREM a A Št (总 额 )。 
SQL SUM() 语法 


SELECT SUM(column_name) FROM table_name 


SQL SUM() 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_Id OrderDate OrderPrice Customer 
1 2008/12/29 1000 Bush 
2 2008/11/23 1600 Carter 
3 2008/10/05 700 Bush 
4 2008/09/28 300 Bush 
5 2008/08/06 2000 Adams 
6 2008/07/21 100 Carter 


现在 ， 我 们 希望 查找 "OrderPrice" 字段 的 总 数 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT SUM(OrderPrice) AS OrderTotal FROM Orders 
结果 集 类 似 这 样 : 


OrderTotal 
5700 


SQL GROUP BY 语句 


Sit Ee (比如 SUM) 常常 需要 添加 GROUP BY 语句 。 


GROUP BY 语句 


GROUP BY 语句 用 于 结合 合计 男 数 ， 根 据 一 个 或 多 个 列 对 结果 集 进 行 分 组 。 


SQL GROUP BY 语法 


SELECT column_name, aggregate_function(column_name ) 


FROM table name 


WHERE column_name operator value 


GROUP BY column_name 


SQL GROUP BY 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_lId OrderDate 


1 2008/12/29 
2008/11/23 
2008/10/05 
2008/09/28 
2008/08/06 
2008/07/21 


O oO FPF OO N 


OrderPrice 
1000 
1600 
700 
300 
2000 
100 


现在 ， 我 们 希望 查找 每 个 客户 的 总 金额 (总 订单 ) 。 
我 们 想 要 使 用 GROUP BY 语句 对 客户 进行 组 合 。 


我 们 使 用 下 列 SQL 语句 : 


SELECT Customer, SUM(COrderPrice) FROM Orders 


GROUP BY Customer 


结果 集 类 似 这 样 : 


Customer 
Bush 
Carter 
Bush 
Bush 
Adams 


Carter 


Customer SUM(OrderPrice) 


Bush 2000 
Carter 1700 
Adams 2000 


很 棒 吧 ， 对 不 对 ? 
让 我 们 看 一 下 如 果 省 略 GROUP BY 会 出 现 什 么 情况 : 


SELECT Customer, SUM(COrderPrice) FROM Orders 


结果 集 类 似 这 样 : 
Customer SUM(OrderPrice) 
Bush 5700 
Carter 5700 
Bush 5700 
Bush 5700 
Adams 5700 
Carter 5700 


上 面 的 结果 集 不 是 我 们 需要 的 。 


那么 为 什么 不 能 使 用 上 面 这 条 SELECT 语句 呢 ?解释 如 下 : 上 面 的 SELECT 语句 
指定 了 两 列 (Customer 和 SUM(OrderPrice)) 。"SUM(OrderPrice)" 返回 一 个 单独 
的 值 ("OrderPrice" 列 的 总 计 ) ， 而 "Customer" 返回 6 ha (每 个 值 对 应 
"Orders" 表 中 的 每 一 行 ) 。 因 此 ， 我 们 得 不 到 正确 的 结果 。 不 过 ， 您 已 经 看 到 了 ， 
GROUP BY 语句 解决 了 这 个 问题 。 


GROUP BY 一 个 以 上 的 列 


我 们 也 可 以 对 一 个 以 上 的 列 应 用 GROUP BY 语句 ， 就 像 这 样 : 


SELECT Customer, OrderDate, SUM(OrderPrice) FROM Orders 
GROUP BY Customer, OrderDate 


SQL HAVING 子 句 


HAVING 子 句 


在 SQL 中 增加 HAVING 子 句 原因 是 ，WHERE 关键 字 无 法 与 合计 加 数 一起 使 用 。 


SQL HAVING 语法 


SELECT column_name, aggregate_function(column_name) 
FROM table_name 

WHERE column_name operator value 

GROUP BY column_name 

HAVING aggregate_function(column_name) operator value 


SQL HAVING 实例 


我 们 拥有 下 面 这 个 "Orders" K : 


O_Id OrderDate OrderPrice Customer 
1 2008/12/29 1000 Bush 
2 2008/11/23 1600 Carter 
3 2008/10/05 700 Bush 
4 2008/09/28 300 Bush 
5 2008/08/06 2000 Adams 
6 2008/07/21 100 Carter 


现在 ， 我 们 希望 查找 订单 总 金额 少 于 2000 的 客户 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT Customer, SUM(COrderPrice) FROM Orders 
GROUP BY Customer 
HAVING SUM(OrderPrice)<2000 


结果 集 类 似 : 


Customer SUM(OrderPrice) 
Carter 1700 


现在 我 们 希望 查找 客户 "Bush" X "Adams" 拥有 超过 1500 的 订单 总 金额 。 
我 们 在 SQL 语句 中 增加 了 一 个 普通 的 WHERE 子 名 : 

SELECT Customer, SUM(COrderPrice) FROM Orders 

WHERE Customer='Bush' OR Customer='Adams' 


GROUP BY Customer 
HAVING SUM(OrderPrice)>1500 


Customer SUM(OrderPrice) 
Bush 2000 
Adams 2000 


V7, 


SQL UCASE() 函数 


UCASE() 函数 
UCASE 函数 把 字段 的 值 转换 为 大 写 。 
SQL UCASE() 语法 


SELECT UCASE(column_name) FROM table_name 


SQL UCASE() 实例 


我 们 拥有 下 面 这 个 "Persons" X : 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


现在 ， 我 们 希望 选取 "LastName" $0 "FirstName" 列 的 内 容 ， 然 后 把 "LastName" 
列 转换 为 大 写 。 


我 们 使 用 如 下 SQL 语句 : 


SELECT UCASE(LastName) as LastName,FirstName FROM Persons 


结果 集 类 似 这 样 : 
LastName FirstName 
ADAMS John 
BUSH George 


CARTER Thomas 


A 


SQL LCASE() 函数 
LCASE() 函数 

LCASE 函数 把 字段 的 值 转换 为 小 写 。 
SQL LCASE() 语法 


SELECT LCASE(column_name) FROM table_name 


SQL LCASE() 实例 


我 们 拥有 下 面 这 个 "Persons" X : 


Id LastName FirstName Address City 
1 Adams John Oxford Street London 

2 Bush George Fifth Avenue New York 
3 Carter Thomas Changan Street Beijing 


现在 ， 我 们 希望 选取 "LastName" #1 "FirstName" 列 的 内 容 ， 然 后 把 "LastName" 
列 转换 为 小 写 。 


我 们 使 用 如 下 SQL 语句 : 


SELECT LCASE(LastName) as LastName,FirstName FROM Persons 


结果 集 类 似 这 样 : 
LastName FirstName 
adams John 
bush George 


carter Thomas 


SQL MID() 西数 


名 


MID() 函数 


MID 函数 用 于 从 文本 字段 中 提取 字符 。 


SQL MID() 语法 


SELECT MID(column_name,start[,length]) FROM table_name 


参数 描述 
column_name ”必需 。 要 提取 字符 的 字段 。 
start 必需 。 规 定 开始 位 置 (起 始 值 是 1) 。 


length 本 


SQL MID() 实例 


我 们 拥有 下 面 这 个 "Persons" K : 


Id LastName FirstName Address 
1 Adams John Oxford Street 

2 Bush George Fifth Avenue 

3 Carter Thomas Changan Street 


现在 ， 我 们 希望 从 "City" 列 中 提取 前 3 个 字符 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT MID(City,1,3) as SmallCity FROM Persons 


结果 集 类 似 这 术 


规 
可 选 。 要 返回 的 字符 数 。 如 果 省 略 ， 则 MID() K 


EB] Fl RIT 


City 
London 
New York 
Beijing 
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SmallCity 
Lon 
New 


Bei 


SQL MID() EX 


129 


eA 


SQL LEN() 函数 


A 


LEN() 西数 
LEN 画 数 返回 文本 字段 中 值 的 长 度 。 
SQL LEN() 语法 


SELECT LEN(column_name) FROM table_name 


SQL LEN() 实例 


我 们 拥有 下 面 这 个 "Persons" X : 


Id LastName FirstName Address 
1 Adams John Oxford Street 

2 Bush George Fifth Avenue 

3 Carter Thomas Changan Street 


现在 ， 我 们 希望 取得 "City" 列 中 值 的 长 度 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT LEN(City) as LengthOfCity FROM Persons 


结果 集 类 似 这 样 : 


LengthOfCity 


City 
London 
New York 
Beijing 


SQL ROUND() 函数 
ROUND() 西数 
ROUND 画 数 用 于 把 数值 字段 舍 入 为 指定 的 小 数位 数 。 


SQL ROUND() 语法 


SELECT ROUND(column_name, decimals) FROM table_name 


参数 描述 
column_name 必需 。 要 伟人 的 字段 。 
decimals 必需 。 规 定 要 返回 的 小 数位 数 。 


SQL ROUND() 实例 


我 们 拥有 下 面 这 个 "Products" 表 : 


Prod_Id ProductName Unit UnitPrice 
1 gold 1000 g 32.35 
2 silver 1000 g 11.56 
3 copper 1000 g 6.85 
现在 ， 我 们 希望 把 名 称 和 价格 舍 入 为 最 接近 的 整数 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT ProductName, ROUND(UnitPrice,0) as UnitPrice FROM Products 
es 


结果 集 类 似 这 样 : 


W3School Sql 教程 


ProductName 
gold 
silver 


copper 


SQL ROUND() 2X 


32 
12 


UnitPrice 


132 


eA 


SQL NOW() 2X 


A 


NOW() 函数 

NOW 画 数 返回 当前 的 日 期 和 时 间 。 

提示 : 如 果 您 在 使 用 Sql Server 数据 库 ， 请 使 用 getdate() 函数 来 获得 当前 的 日 期 
时 间 o 

SQL NOW() 语法 


SELECT NOW() FROM table_name 


SQL NOW() 实例 


我 们 拥有 下 面 这 个 "Products" X : 


Prod_Id ProductName Unit UnitPrice 
1 gold 1000 g 32.35 
2 silver 1000 g 11.56 
3 copper 1000 g 6.85 


现在 ， 我 们 希望 显示 当天 的 日 期 所 对 应 的 名 称 和 价格 。 
我 们 使 用 如 下 SQL 语句 : 


SELECT ProductName, UnitPrice, Now() as PerDate FROM Products 


结果 集 类 似 这 样 : 
ProductName UnitPrice PerDate 
gold 32.35 12/29/2008 11:36:05 AM 
silver 11.56 12/29/2008 11:36:05 AM 


copper 6.85 12/29/2008 11:36:05 AM 


SQL FORMAT() 函数 
FORMAT() 2X 

FORMAT 函数 用 于 对 字段 的 显示 进行 格式 化 。 
SQL FORMAT() 语法 


SELECT FORMAT(column_name, format) FROM table_name 


参数 描述 
column_name 必需 。 要 格式 化 的 字段 。 
format 必需 。 规 定格 式 。 


SQL FORMAT() 实例 


我 们 拥有 下 面 这 个 "Products" 表 : 


Prod_Id ProductName Unit UnitPrice 
1 gold 1000 g 32.35 
2 silver 1000 g 11.56 
3 copper 1000 g 6.85 


现在 ， 我 们 希望 显示 每 天 日 期 所 对 应 的 名 称 和 价格 (日 期 的 显示 格式 是 "YYYY- 
MM-DD") 。 


我 们 使 用 如 下 SQL 语句 : 


SELECT ProductName, UnitPrice, FORMAT(Now(), 'YYYY-MM-DD') as PerDat 
FROM Products 


eee 





结果 集 类 似 这 样 : 


ProductName UnitPrice PerDate 
gold 32.35 12/29/2008 
silver 11.56 12/29/2008 
copper 6.85 12/29/2008 


SQL 快速 参考 


来 自 W3School 的 SQL 快速 参考 。 可 以 打印 它 ， 以 备 日 常 使 用 。 


SQL 语句 
语句 语法 
AND/OR SELECT column_name(s) FROM table_name WHERE 


ALTER TABLE (add 
column) 


ALTER TABLE 
(drop column) 


AS (alias for 
column) 


AS (alias for table) 


BETWEEN 


CREATE 
DATABASE 


CREATE INDEX 


CREATE TABLE 


CREATE UNIQUE 
INDEX 


CREATE VIEW 


DELETE FROM 


DROP DATABASE 
DROP INDEX 
DROP TABLE 


condition AND|OR condition 


ALTER TABLE table_name ADD column_name 
datatype 


ALTER TABLE table_name DROP COLUMN 
column_name 


SELECT column_name AS column_alias FROM 
table_name 


SELECT column_name FROM table_name AS 
table_alias 


SELECT column_name(s) FROM table_name WHERE 
column_name BETWEEN value1 AND value2 
CREATE DATABASE database_name 


CREATE INDEX index_name ON table_name 
(column_name) 


CREATE TABLE table_name ( column_name1 
data_type, column_namez2 data type, ....... ) 


CREATE UNIQUE INDEX index_name ON table_name 
(column_name) 


CREATE VIEW view_name AS SELECT 
column_name(s) FROM table_name WHERE condition 


DELETE FROM tablename (Note: Deletes the entire 
table!!) or DELETE FROM table_name WHERE 
condition 


DROP DATABASE database_name 
DROP INDEX table_name.index_name 
DROP TABLE table_name 


GROUP BY 


HAVING 


IN 


INSERT INTO 


LIKE 


ORDER BY 


SELECT 
SELECT * 


SELECT DISTINCT 


SELECT INTO 
(used to create 
backup copies of 
tables) 


TRUNCATE TABLE 
(deletes only the 
data inside the 
table) 


UPDATE 


WHERE 


SELECT column_name1,SUM(column_name2) FROM 
table_name GROUP BY column_name1 


SELECT column_name1,SUM(column_name2) FROM 
table_name GROUP BY column_name1 HAVING 
SUM(column_name2) condition value 


SELECT column_name(s) FROM table_name WHERE 
column_name IN (value1,value2,..) 


INSERT INTO tablename VALUES (value1, value2,....) 


_or INSERT INTO table_name (column_name1, 


column_namez2,...) VALUES (value1, valuez2.....) 


SELECT column_name(s) FROM table_name WHERE 
column_name LIKE pattern 


SELECT column_name(s) FROM table_name ORDER 
BY column_name [ASC|DESC] 


SELECT column_name(s) FROM table_name 
SELECT * FROM table_name 
SELECT DISTINCT column_name(s) FROM 


table_name 


SELECT * INTO newtable_name FROM 
original_table_name _or SELECT column_name(s) 
INTO new_table_name FROM original_table_name 


TRUNCATE TABLE table_name 


UPDATE table_name SET column_name=new_value |, 
column_name=new_value] WHERE 
column_name=some_value 


SELECT column_name(s) FROM table_name WHERE 
condition 


SQLite 基础 


SQLite 简介 


本 教程 帮助 您 了 解 什 么 是 SQLite， 


它 与 SQL 之 间 的 不 同 ， 为 什么 需要 它 ， 以 及 它 


的 应 用 程序 数据 库 处 理 方式 。 


SQLite 是 一 个 软件 库 ， 实 现 了 自给 自足 的 、 无 服务 器 的 、 雪 配置 的 、 事 务 性 的 SQL 
数据 库 引 擎 。SQLite 是 一 个 增长 最 快 的 数据 库 引 擎 ， 这 是 在 普及 方面 的 增长 ， 和 与 它 


的 尺寸 大 小 无 关 。SQLite 源 代码 不 受 版 权限 制 。 


什么 是 SQLite ? 


SQLite 是 一 个 进程 内 的 库 ， 实 现 了 自给 自足 的 、 无 服务 器 的 、 雳 配置 的 、 事 务 性 的 
SQL 数据 库 引 擎 。 CH 是 一 个 零 配置 的 数据 库 ， 这 意味 着 与 其 他 数据 库 一 


要 在 系统 中 配置。 


就 像 其 他 数据 库 ，SQLite 引擎 不 是 一 个 独立 的 进程 ， 可 以 按 应 用 程序 需 


或 动态 连接 。SQLite 直接 访问 其 存储 文件 。 


为 什么 要 用 SQLite ? 


不 需要 一 个 单独 的 服务 器 进程 或 操作 的 系统 (无 服务 器 的 ) 。 
SQLite 不 需要 配置 ， 这 意味 着 不 需要 安装 或 管理 。 
一 个 完整 的 SQLite 数据 库 是 存储 在 一 个 单一 的 跨 平 台 的 磁盘 文件 。 


SQLite 是 非常 小 的 ， 是 轻 量 级 的 ， 完 全 配置 时 小 于 400KiB， 省 上 略 可 选 功能 配 
置 时 小 于 250KiB。 


SQLite 是 自给 自足 的 ， 这 意味 着 不 需要 任何 外 部 的 依赖 。 

SQLite 事务 是 完全 兼容 ACID 的 ， 人 允许 从 多 个 进程 或 线程 安全 访问 。 
SQLite 支持 SQL92 (SQL2) 标准 的 大 多 数 查询 语言 的 功能 。 
SQLite 使 用 ANSI-C 编写 的 ， 并 提供 了 简单 和 易于 使 用 的 APl。 


SQLite 可 在 UNIX (Linux, Mac OS-X, Android, iOS) 和 Windows (Win32, 
WinCE, WinRT) 中 运行 。 


历史 


1 
2. 


. 2000 -- D. Richard Hipp 设计 SQLite 是 为 了 不 需要 管理 即 可 操作 程序 。 


2000 -- 在 八 月 ，SQLite1.0 发 布 GNU 数据 库 管 理 器 (GNU Database 
Manager) 。 


样 ， 您 不 需 


aR PETIT BRA 


3. 2011 -- Hipp 宣布 ， 向 SQLite DB 添加 UNQI 接口 ， 开 发 UNQLite (面向 文档 
的 数据 库 ) 。 


SQLite 局 限 性 
在 SQLite 中 ，SQL92 不 支持 的 特性 如 下 所 示 : 
特性 描述 
RIGHT 
OUTER 只 实现 了 LEFT OUTER JOIN。 
JOIN 
FULL 
OUTER 只 实现 了 LEFT OUTER JOIN。 
JOIN 
ALTER 支持 RENAME TABLE 和 ALTER TABLE 的 ADD COLUMN 
TABLE variants 命令 ， 不 支持 DROP COLUMN, ALTER COLUMN, 
ADD CONSTRAINT. 
Trigger 支持 FOR EACH ROW 触发 器 ,但 不 支持 FOR EACH 
支持 STATEMENT 触发 器 。 
VIEWs 在 SQLite 中 ， 视 图 是 只 读 的 。 您 不 可 以 在 视图 上 执行 DELETE、 
INSERT 3 UPDATE 语句 。 
GRANT 
和 可 以 应 用 的 唯一 的 访问 权限 是 底层 操作 系统 的 正常 文件 访问 权限 。 
REVOKE 
SQLite MS 


与 关系 数据 库 进 行 交 互 的 标准 SQLite 命令 类 似 于 SQL, 命令 包 括 CREATE, 
ERD 


SELECT, 


INSERT, UPDATE, DELETE 和 DROP。 这 


基于 它们 的 操作 性 


质 可 分 为 以 下 几 种 : 


DDL - 数据 定义 语言 


AS 
AR T3 


CREATE 


ALTER 
DROP 


描述 
创建 一 个 新 的 表 ， 一 个 表 的 视图 ， 或 者 数据 库 中 的 其 他 对 象 。 
修改 数据 库 中 的 某 个 已 有 的 数据 库 对 象 ， 比 如 一 个 表 。 
除 整 个 表 ， 或 者 表 的 视图 ， 或 者 数据 库 中 的 其 他 对 象 。 


DML - 数据 操作 语言 


命 兮 
INSERT 创建 一 条 记录 。 
UPDATE 修改 记录 。 
DELETE 删除 记录 。 
DQL - 数据 查询 语言 
ay 描述 


SELECT 从 一 个 或 多 个 表 中 检索 某 些 记录 。 


it 


学 


SQLite 安装 


SQLite 的 一 个 重要 的 特性 是 雳 配置 的 ， 这 意味 着 不 需要 复杂 的 安装 或 管理 。 本 章 将 
讲解 Windows、Linux 和 Mac OS X 上 的 安装 设置 。 


在 Windows 上 安装 SQLite 


e 请 访问 SQLite 下 载 页 面 ， 从 Windows 区 下 载 预 编译 的 二 进 制 文件 。 
。 您 需要 下 载 sqlite-shell-win32-*.zip 和 sqlite-dll-win32-*.zip 压缩 文件 。 


。 创建 文件 夹 C:\v>sqlite， 并 在 此 文件 夹 下 解压 上 面 两 个 压缩 文件 ， 将 得 到 
sqlite3.def、sqlite3.dll 和 sqlite3.exe 文件 。 


e 添加 C:\>sqlite 到 PATH 环境 变量 ， 最 后 在 命令 提示 符 下 ， 使 用 sqlite3 命 
令 ， 将 显示 如 下 结果 。 


C:\>sqlite3 

SQLite version 3.7.15.2 2013-01-09 11:53:05 
Enter ".help" for instructions 

Enter SQL statements terminated with a ";" 
sqlite> 


在 Linux 上 安装 SQLite 


目前 ， 几 乎 所 有 版 本 的 Linux 操作 系统 都 附带 SQLite。 所 以 ， 只 要 使 用 下 面 的 命令 
来 检查 您 的 机 器 上 是 否 已 经 安装 了 SQLite. 


$sqlite3 

SQLite version 3.7.15.2 2013-01-09 11:53:05 
Enter ".help" for instructions 

Enter SQL statements terminated with a ";" 
sqlite> 


如 果 没 有 看 到 上 面 的 结果 ， 那 么 就 意味 着 没有 在 Linux 机 器 上 安装 SQLite。 因 此 ， 
让 我 们 按照 下 面 的 步骤 安装 SQLite : 


e 请 访问 SQLite 下 载 页 面 ， 从 源 代码 区 下 载 sqlite-autoconf-*.tar.gz。 
。 步骤 如 下 : 


$tar xvfz Sqlite-autoconf-3071502 .tar ,gz 
$cd sqlite-autoconf -3071502 

$./configure --prefix=/usr/local 

$make 

$make install 


上 述 步骤 将 在 Linux 机 器 上 安装 SQLite， 您 可 以 按照 上 述 讲解 的 进行 验证 。 


在 Mac OS X 上 安装 SQLite 


最 新 版 本 的 Mac OS X 会 预 安 装 SQLite， 但 是 如 果 没 有 可 用 的 安装 ， 只 需 按照 如 
下 步 又 进行 : 


e 请 访问 SQLite 下 载 页 面 ， 从 源 代码 区 下 载 sqlite-autoconf-*.tar.gz。 
o 步骤 如 下 : 


$tar xvfz Sqlite-autoconf-3071502 .tar ,gz 
$cd sqlite-autoconf -3071502 

$./configure --prefix=/usr/local 

$make 

$make install 


上 述 步 骤 将 在 Mac OS X 机 器 上 安装 SQLite， 您 可 以 使 用 下 列 命令 进行 验证 : 


$sqlite3 

SQLite version 3.7.15.2 2013-01-09 11:53:05 
Enter ".help" for instructions 

Enter SQL statements terminated with a ";" 
sqlite> 


最 后 ， 在 SQLite 命令 提示 符 下 ， 使 用 SQLite 命令 做 练习 。 


SQLite MS 

本 章 将 向 您 讲解 SQLite 编程 人 员 所 使 用 的 简单 却 有 用 的 命令 。 些 

SQLite 的 点 命令 ， 这 些 命令 的 不 同 之 处 在 于 它们 不 以 分 号 (;) AR. 

让 我 们 在 命令 提 : 示 符 下 键入 一 个 简单 的 sqlite3 MA, E SQLite 命令 提示 符 下 ， 您 
可 以 使 用 各 种 SQLite MT. 


$sqlite3 
SQLite version 3.3.6 


Enter ".help" for instructions 


sqlite> 


如 需 获 取 可 用 的 点 命令 的 清 羊 ， 可 以 在 任何 时 候 输 入 "help"。 例 如 : 


sqlite>.help 


TABLE 


上 面 的 命令 会 显示 各 种 重要 的 SQLite 点 命令 的 列表 ， 如 下 所 示 : 
.backu ? Ew nll B 
R a 44) DB 数据 库 (默认 是 "main") 到 FILE 文件 。 
Dail 发 生 错 误 后 停止 。 默 认为 OFF 
ONIOFF 首 误 后 T o ANL y o 
databases 列 出 附加 数据 库 的 名 称 和 文件 。 
.dump ? 以 SQL 文本 格式 转 储 数据 库 。 如 果 指 定 了 TABLE 表 ， 则 只 转 
TABLE? 储 匹 配 LIKE 模式 的 TABLE 表 。 
.echo ree 
ONIOFF 开局 或 关闭 echo 命令 。 
.exit 退出 SQLite 提示 符 。 
.explain 开启 或 关闭 适合 于 EXPLAIN 的 输出 模式 。 如 果 没 有 带 参数 ， 则 
ON|OFF X EXPLAIN on, RFF EXPLAIN. 
.header(s) E ig L R a = 
ONIOFF 开启 或 关闭 头 部 显示 。 
.help 显示 消息 。 
.Import 
FILE 导入 来 自 FILE 文件 的 数据 到 TABLE RH, 


indices ? 
TABLE? 


load FILE 
?ENTRY? 


log 
FILE |off 


.mode 
MODE 


-nullvalue 
STRING 


output 
FILENAME 


.output 
stdout 


.print 
STRING... 


.prompt 
MAIN 
CONTINUE 


quit 


read 
FILENAME 


schema ? 
TABLE? 


.separator 
STRING 


.Show 


.Stats 
ONIOFF 


.tables ? 
PATTERN? 


timeout 
MS 


.width NUM 


显示 所 有 索引 的 名 称 。 如 果 指 定 了 TABLE 表 ， 则 只 显示 匹配 
LIKE 模式 的 TABLE 表 的 索引 。 


加 载 一 个 扩展 库 。 


开启 或 关闭 日 志 。FILE 文件 可 以 是 stderr (标准 错 
误 ) /stdout (标准 输出 ) o 


设置 输出 模式 ，MODE 可 以 是 下 列 之 一 : csv 到 号 分 隔 的 值 
column 4 xt #899!) html HTML 的 <table> 代码 insert 
TABLE 表 的 SQL 插入 (insert) 语句 line 每 行 一 个 值 list 由 
.Separator 字符 串 分 隔 的 值 tabs 由 Tab 分 隔 的 值 tel TCL 列表 
元 素 


在 NULL 值 的 地 方 输出 STRING 字符 串 。 
发 送 输 出 到 FILENAME 文件 。 
发 送 输出 到 屏幕 。 


逐 字 地 输出 STRING 字符 串 。 


蔡 换 标准 提示 符 。 


退出 SQLite 提示 符 。 
执行 FILENAME 文件 中 的 SQL。 


显示 CREATE 724), SORE S TABLE 表 ， 则 只 显示 匹配 
LIKE 模式 的 TABLE 表 。 


改变 输出 模式 和 import 所 使 用 的 分 隔 符 。 
显示 各 种 设置 的 当前 值 。 
开启 或 关闭 统计 。 

列 出 匹配 LIKE 模式 的 表 的 名 称 。 


尝试 打开 锁定 的 表 MS 微 秒 。 


NUM 


timer Tabi mi on 、| E 
ONIOFF 开启 或 关闭 CPU TERY Ze. 
让 我 们 尝试 使 用 .show 命令 ， 来 查看 SQLite 命令 提示 符 的 默认 设置 。 


sqlite>.show 
echo: off 
explain: off 
headers: off 
mode: column 
nullvalue: "" 
output: stdout 
separator: "|" 
width: 
sqlite> 


> 确保 sqlite> 提示 符 与 点 命令 之 间 没有 空格 ， 否 则 将 无 法 正常 工作 。 


格式 化 输出 
您 可 以 使 用 下 列 的 点 命 全 来 格式 化 输出 为 本 教程 下 面 所 列 出 的 格式 : 


sqlite>.header on 
sqlite>.mode column 
sqlite>.timer on 
sqlite> 


上 面 设置 将 产生 如 下 格式 的 输出 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 
CPU Time: user 0.000000 sys 0.000000 


sqlite_master 表格 


主 表 中 保存 数据 库 表 的 关键 信息 ， 并 把 它 命 名 为 sqlite_master。 如 要 查看 表 概 
要 ， 可 按 如 下 操作 : 


sqlite>.schema sqlite_master 


这 将 产生 如 下 结果 : 


CREATE TABLE sqlite master ( 
type text, 

name text, 

tbl_name text, 

rootpage integer, 

sql text 

); 


SQLite 语法 


SQLite 是 遵循 一 套 独特 的 称 为 语法 的 规则 和 准则 。 本 教程 列 出 了 所 有 基本 的 
SQLite 语法 ， 向 您 提供 了 一 个 SQLite 快速 入 门 。 


大 小 写 敏 感性 


有 个 重要 的 点 值得 注意 ，SQLite 是 不 区 分 大 小 写 的 ， 但 也 有 一 些 命令 是 大 小 写 敏 感 
的 ， 比 如 GLOB 和 glob 在 SQLite 的 语句 中 有 不 同 的 含义 。 





注释 


SQLite 注释 是 附加 的 注释 ， 可 以 在 SQLite 代码 中 添加 注释 以 增加 其 可 读 性 ， 他 们 
可 以 出 现在 任何 空白 处 ， 包 括 在 表达 式 内 和 其 他 SQL 语句 的 中 间 ， 但 它们 不 能 启 


o 


SQL 注释 以 两 个 连续 的 "-" 字符 (ASCII 0x2d) 开始 ， 并 扩展 至 下 一 个 换行 符 
(ASCII 0x0a) 或 直到 输入 结束 ， 以 先 到 者 为 准 。 


您 也 可 以 使 用 C 风格 的 注释 ， 以 "/" 开始， 并 扩展 至 下 一 个 "字符 对 或 直到 输入 结 
束 ， 以 先 到 者 为 准 。C 庚 哥 的 注释 可 以 跨越 多 行 。 


sqlite>.help -- This is a single line comment 


SQLite 语句 


所 有 的 SQLite 语句 可 以 以 任何 关键 字 开 始 ， 如 SELECT. INSERT, UPDATE, 
DELETE, ALTER, DROP 等 ， 所 有 的 语句 以 分 号 (;) 结束 。 


SQLite ANALYZE 语句 : 


ANALYZE; 
or 
ANALYZE database_name; 


or 
ANALYZE database_name.table_name; 


SQLite AND/OR #4 : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE CONDITION-1 {AND|OR} CONDITION-2; 


SQLite ALTER TABLE 语句 : 


ALTER TABLE table_name ADD COLUMN column_def...; 


SQLite ALTER TABLE 324) (Rename) 


ALTER TABLE table_name RENAME TO new_table_name; 


SQLite ATTACH DATABASE 语句 : 


ATTACH DATABASE 'DatabaseName' As 'Alias-Name'; 


SQLite BEGIN TRANSACTION 语句 : 


BEGIN; 
or 
BEGIN EXCLUSIVE TRANSACTION; 


SQLite BETWEEN Ff% : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE column_name BETWEEN val-1 AND val-2; 


SQLite COMMIT 4% : 


COMMIT; 


SQLite CREATE INDEX 语句 : 


CREATE INDEX index_name 
ON table_name ( column_name COLLATE NOCASE ); 


SQLite CREATE UNIQUE INDEX 语句 : 


CREATE UNIQUE INDEX index_name 
ON table_name ( columni, column2,...columnN); 


SQLite CREATE TABLE 语句 : 


CREATE TABLE table_name( 
columni datatype, 
column2 datatype, 
column3 datatype, 
columnN datatype, 
PRIMARY KEY( one or more columns ) 


); 


SQLite CREATE TRIGGER 语句 : 


CREATE TRIGGER database_name.trigger_name 
BEFORE INSERT ON table_name FOR EACH ROW 
BEGIN 

stmt1; 

stmt2; 


END; 


SQLite CREATE VIEW 语句 : 


CREATE VIEW database_name.view_name AS 
SELECT statement....; 


SQLite CREATE VIRTUAL TABLE 语句 : 


CREATE VIRTUAL TABLE database_name.table_name USING weblog( access 


or 
CREATE VIRTUAL TABLE database_name.table_name USING fts3( ); 








‘| 





SQLite COMMIT TRANSACTION 语句 : 


COMMIT; 


SQLite COUNT F% : 


SELECT COUNT(column_name) 
FROM table_name 
WHERE CONDITION; 


SQLite DELETE 语句 : 


DELETE FROM table name 
WHERE {CONDITION}; 


SQLite DETACH DATABASE 语句 : 


DETACH DATABASE 'Alias-Name'; 


SQLite DISTINCT #4) : 


SELECT DISTINCT columni, column2....columnN 
FROM table_name; 


SQLite DROP INDEX 语句 : 


DROP INDEX database_name. index_name; 


SQLite DROP TABLE 语句 : 


DROP TABLE database_name.table_name; 


SQLite DROP VIEW 语句 : 


DROP INDEX database_name.view_name; 


SQLite DROP TRIGGER 语句 : 


DROP INDEX database_name.trigger_name; 


SQLite EXISTS F% : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE column_name EXISTS (SELECT * FROM table_name ); 


SQLite EXPLAIN 语句 : 


EXPLAIN INSERT statement...,; 
or 
EXPLAIN QUERY PLAN SELECT statement...; 


SQLite GLOB F% : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE column_name GLOB { PATTERN }; 


SQLite GROUP BY £4) : 


SELECT SUM(column_name) 
FROM table_name 
WHERE CONDITION 

GROUP BY column_name; 


SQLite HAVING +4) : 


SELECT SUM(column_name) 

FROM table_name 

WHERE CONDITION 

GROUP BY column_name 

HAVING (arithematic function condition); 


SQLite INSERT INTO 语句 : 


INSERT INTO table _name( columni, column2....columnN) 
VALUES ( valuei, value2....valueN); 


SQLite IN #4 : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE column_name IN (val-1, val-2,...val-N); 


SQLite Like F% : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE column_name LIKE { PATTERN }; 


SQLite NOT IN 子 句 : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE column_name NOT IN (val-1, val-2,...val-N); 


SQLite ORDER BY 子 句 : 


SELECT columni, column2....columnN 
FROM table_name 

WHERE CONDITION 

ORDER BY column_name {ASC|DESC}; 


SQLite PRAGMA 语句 : 


PRAGMA pragma_name; 
For example: 


PRAGMA page_size; 
PRAGMA cache_size = 1024; 
PRAGMA table_info(table_name); 


SQLite RELEASE SAVEPOINT 语句 : 


RELEASE savepoint_name; 


SQLite REINDEX 语句 : 


REINDEX collation_name; 
REINDEX database_name. index_name; 
REINDEX database_name.table_name; 


SQLite ROLLBACK 语句 : 


ROLLBACK; 
or 
ROLLBACK TO SAVEPOINT savepoint_name; 


SQLite SAVEPOINT 语句 : 


SAVEPOINT savepoint_name; 


SQLite SELECT 语句 : 


SELECT columni, column2....columnN 
FROM table_name; 


SQLite UPDATE 语句 : 


UPDATE table_name 
SET columni = value1, column2 = value2....columnN=valueN 
[ WHERE CONDITION ]; 


SQLite VACUUM 语句 : 


VACUUM; 


SQLite WHERE +4) : 


SELECT columni, column2....columnN 
FROM table_name 
WHERE CONDITION; 


SQLite 数据 类 型 
SQLite 数据 关 型 是 一 个 用 来 指定 任何 对 象 的 数据 关 型 的 属性 。SQLite 中 的 每 一 
列 ， 每 个 变量 和 表达 式 都 有 相关 的 数据 类 型 。 


您 可 以 在 创建 表 的 同时 使 用 这 些 数 据 类 型 。SQLite 使 用 一 个 更 普通 的 动态 类 型 系 
统 。 在 SQLite 中 ， 值 的 数据 类 型 与 值 本 身 是 相关 的 ， 而 不 是 与 它 的 容器 相关 。 


SQLite 存储 类 
每 个 存储 在 SQLite 数据 库 中 的 值 都 具有 以 下 存储 类 之 一 : 
存储 类 描述 
NULL 值 是 一 个 NULL 值 。 
INTEGER 值 是 一 个 带 符号 的 整数 ， 根据 值 的 大 小 存储 在 1、2、3、4、6 或 
8 字 节 中 。 
REAL 值 是 一 个 浮 点 值 ， 存 储 为 8 字 节 的 IEEE 浮 点 数字 。 
TEXT 值 是 一 个 文本 字符 串 ， 使 用 数据 库 编码 (UTF-8、UTF-16BE 或 
UTF-16LE) 存储 。 
BLOB 值 是 一 个 blob 数据 ， 完 全 根据 它 的 输入 存储 。 


SQLite 的 存储 类 各 微 比 数据 类 型 更 普 志 。INTEGER 存储 类 ， 例 如 ， 包 含 6 种 不 同 
的 不 同 长 度 的 整数 数据 类 型 。 


SQLite Affinity 类 型 


SQLite 支持 列 上 的 类 型 affinity 概念 。 任 何 列 仍然 可 以 存储 任何 类 型 的 数据 ， 但 列 
的 首选 存储 类 是 它 的 affinity。 在 SQLite3 数据 库 中 ， 每 个 表 的 列 分 配 为 以 下 类 型 
BY affinity 之 一 : 


Affinity 
TEXT 
NUMERIC 


INTEGER 


REAL 


NONE 


该 列 使 用 存储 类 NULL, TEXT 或 BLOB 存储 所 有 数据 。 

该 列 可 以 包含 使 用 所 有 五 个 存储 类 的 值 。 

与 带 有 NUMERIC affinity 的 列 相 同 ， 在 CAST 表达 式 中 带 有 及 
常 。 

与 带 有 NUMERIC affinity 的 列 相似 ， 不 同 的 是 ， 它 会 强制 把 整数 
值 转换 为 浮 点 表示 。 

带 有 affinity NONE 的 列 ， 不 会 优先 使 用 哪个 存储 类 ， 人 也 不 会 党 试 
把 数据 从 一 个 存储 类 强制 转换 为 另 一 个 存储 类 。 


SQLite Affinity 及 类 型 名 称 
下 表 列 出 了 当 创 建 SQLite3 表 时 可 使 用 的 各 种 数据 类 型 名 称 ， 同 时 也 显示 了 相应 的 


应 用 Affinity : 
数据 类 型 Affinity 

INT INTEGER TINYINT SMALLINT MEDIUMINT BIGINT INTEGER 
UNSIGNED BIG INT INT2 INT8 
CHARACTER(20) VARCHAR(255) VARYING 
CHARACTER(255) NCHAR(55) NATIVE CHARACTER(70) TEXT 
NVARCHAR(100) TEXT CLOB 
BLOB "no datatype specified" NONE 
REAL DOUBLE DOUBLE PRECISION FLOAT REAL 
NUMERIC DECIMAL(10,5) BOOLEAN DATE DATETIME NUMERIC 


Boolean 数据 类 型 


SQLite 没有 单独 的 Boolean 存储 类 。 相 反 ， 布 尔 值 被 存储 为 整数 0 (false) 和 


1 (true) 。 


Date 与 Time 数据 类 型 


SQLite 没有 一 个 单独 的 用 于 存储 日 期 和 /或 时 间 的 存储 类 ， 但 SQLite 能 够 把 日 期 和 
时 间 存 储 为 TEXT, REAL 或 INTEGER 值 。 


存储 类 日 期 格式 
TEXT 格式 为 "YYYY-MM-DD HH:MM:SS.SSS" 的 日 期 。 
从 公元 前 4714 年 11 月 24 日 格林 尼 治 时 间 的 正午 开始 算 起 的 天 
数 


o 


REAL 
INTEGER ”从 1970-01-01 00:00:00 UTC 算 起 的 秒 数 。 


AE AY ALM fa ERRAR fs OHA, HETARA AE A HAA ial AR 
自由 转换 不 同 格式 。 


SQLite 创建 数据 库 


SQLite 的 sqlite3 命令 被 用 来 创建 新 的 SQLite 数据 库 。 您 不 需要 任何 特殊 的 权限 
即 可 创建 一 个 数据 。 

语法 

sqlite3 命 使 的 基本 语法 如 下 : 


$sqlite3 DatabaseName.db 
通常 情况 下 ， 数 据 库 名 称 在 RDBMS 内 应 该 是 唯一 的 。 


实例 
如 果 您 想 创建 一 个 新 的 数据 库 <testDB.db>，SQLITE3 语句 如 下 所 示 : 


$sqlite3 testDB.db 

SQLite version 3.7.15.2 2013-01-09 11:53:05 
Enter ".help" for instructions 

Enter SQL statements terminated with a ";" 
sqlite> 


上 面 的 命令 将 在 当前 目录 下 创建 一 个 文件 testDB.db。 该 文件 将 被 SQLite 引擎 用 
作 数 据 库 。 如 果 您 已 经 注意 到 sqlite3 命令 在 成 功 创建 数据 库 文 件 之 后 ， 将 提供 一 
个 sqlite> 提示 符 。 


一 旦 数据 库 被 创建 ， 您 就 可 以 使 用 SQLite 的 .databases 命令 来 检查 它 是否 在 数据 
库 列 表 中 ， 如 下 所 示 : 


sqlite>.databases 
seq name file 


0 main /home/sqlite/testDB.db 


您 可 以 使 用 SQLite .quit 命令 退出 sqlite 提示 符 ， 如 下 所 示 : 


sqlite>.quit 
$ 


.dump 命令 


您 可 以 在 命令 提示 符 中 使 用 SQLite dump 点 命令 来 导出 完整 的 数据 库 在 一 个 文本 
文件 中 ， 如 下 所 示 : 


$sqlite3 testDB.db .dump > testDB.sql 


上 面 的 命令 将 转换 整个 testDB.db 数据 库 的 内 容 到 SQLite 的 语句 中 ， 并 将 其 转 储 
到 ASCII 文本 文件 testDB.sql 中 。 您 可 以 通过 简单 的 方式 从 生成 的 testDB.sq| 恢 
复 ， 如 下 所 示 : 


$sqlite3 testDB.db < testDB.sql 


此 时 的 数据 库 是 空 的 ， 一 旦 数据 库 中 有 表 和 数据 ， 您 可 以 尝试 上 述 两 个 程序 。 现 
在 ， 让 我 们 继续 学 习 下 一 章 。 


SQLite 附加 数据 库 
假设 这 样 一 种 情况 ， 当 在 同一 时 间 有 多 个 数据 库 可 用 ， 您 想 使 用 其 中 的 任何 一 个 。 


SQLite 的 ATTACH DTABASE 话 句 是 用 来 选择 一 个 特定 的 数据 库 ， 使 用 该 命令 
后 ， 所 有 的 SQLite 语句 将 在 附加 的 数据 库 下 执行 。 


语法 
SQLite 的 ATTACH DATABASE 语句 的 基本 语法 如 下 : 


ATTACH DATABASE 'DatabaseName' As 'Alias-Name'; 


如 果 数 据 库 尚未 被 创建 ， 上 面 的 命令 将 创建 一 个 数据 库 ， 如 果 数 据 库 已 存在 ， 则 把 
数据 库 文件 名 称 与 逻辑 数据 库 'Alias-Name' 绑 定 在 一 起 。 


实例 


如 果 想 附加 一 个 现 有 的 数据 库 testDB.db， 则 ATTACH DATABASE 语句 将 如 下 所 
小 : 


sqlite> ATTACH DATABASE 'testDB.db' as 'TEST'; 


使 用 SQLite database 命令 来 显示 附加 的 数据 库 。 


sqlite> .database 


seq name file 
0 main /home/sqlite/testDB.db 
2 test /home/sqlite/testDB.db 


数据 库 名 称 main 和 temp RAR A FERGE EAE HH ls RRA GY Bt R 
的 数据 库 。 这 两 个 数据 库 名 称 可 用 于 每 个 数据 库 连 接 ， 且 不 应 该 被 用 于 附加 ， 否 则 


sqlite> ATTACH DATABASE 'testDB.db' as 'TEMP'; 
Error: database TEMP is already in use 
sqlite> ATTACH DATABASE 'testDB.db' as 'main'; 
Error: database TEMP is already in use 


SQLite 分 离 数 据 库 


SQLite 的 DETACH DTABASE 话 句 是 用 来 把 命名 数据 库 从 一 个 数据 库 连 接 分 离 和 
游离 出 来 ， 连 接 是 之 前 使 用 ATTACH 语句 附加 的 。 如 果 同 一 个 数据 库 文 件 已 经 被 
附加 上 多 个 别名 ，DETACH 命令 将 只 断 开 给 定名 称 的 连接 ， 而 其 余 的 仍然 有 效 。 您 
无 法 分 离 main 或 temp 数据 库 。 


> 如 果 数 据 库 是 在 内 存 中 或 者 是 临时 数据 库 ， 则 该 数据 库 将 被 摧毁 ， 上 内容 笃 会 于 


SQLite 的 DETACH DATABASE 'Alias-Name' 语句 的 基本 语法 如 下 : 


DETACH DATABASE 'Alias-Name'; 


在 这 里 ，'Alias-Name' 与 您 之 前 使 用 ATTACH 话 句 附加 数据 库 时 所 用 到 的 别名 相 
同 。 


实例 


假设 在 前 面 的 章节 中 您 已 经 创建 了 一 个 数据 库 ， 并 给 它 附 加 了 'test' 和 
‘currentDB', {#FA .database 命 伟 ， 我 们 可 以 看 到 : 


sqlite>.databases 


seq name file 

0 main /home/sqlite/testDB.db 
2 test /home/sqlite/testDB.db 
3 currentDB /home/sqlite/testDB.db 


现在 ， 让 我 们 尝试 把 'currentDB' 从 testDB.db 中 分 离 出 来 ， 如 下 所 示 : 


sqlite> DETACH DATABASE 'currentDB'; 


现在 ， 如 果 检 查 当 前 附加 的 数据 库 ， 您 会 发 现 ，testDB.db (54 'test' Fl 'main' 保持 
连接 。 


sqlite>.databases 
seq name file 


0 main /home/sqlite/testDB.db 
2 Lest /home/sqlite/testDB.db 


SQLite 创建 表 


SQLite 的 CREATE TABLE 洛 句 用 于 在 任何 给 定 的 数据 库 创建 一 个 新 表 。 创 建 基 
本 表 ， 涉 及 到 命名 表 、 定 义 列 及 每 一 列 的 数据 类 型 。 


语法 
CREATE TABLE 语句 的 基本 语法 如 下 : 


CREATE TABLE database_name.table_name( 
columni datatype PRIMARY KEY(one or more columns), 
column2 datatype, 
column3 datatype, 


columnN datatype, 
); 


CREATE TABLE 是 告诉 数据 库 系 统 创 建 一 个 新 表 的 关键 字 。CREATE TABLE 语句 
后 跟着 表 的 唯一 的 名 称 或 标识 。 您 也 可 以 选择 指定 带 有 table_name 的 
database_name, 


实例 


下 面 是 一 个 实例 ， 它 创建 了 一 个 COMPANY K, ID 作为 主键 ，NOT NULL 的 约束 
表示 在 表 中 创建 纪录 时 这 些 字段 不 能 为 NULL : 


sqlite> CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL 


); 


让 我 们 再 创建 一 个 表 ， 我 们 将 在 随后 章节 的 练习 中 使 用 : 


sqlite> CREATE TABLE DEPARTMENT ( 
ID INT PRIMARY KEY NOT NULL, 
DEPT CHAR(50) NOT NULL, 
EMP_ID INT NOT NULL 


); 


您 可 以 使 用 SQLite 命令 中 的 tables 命令 来 验证 表 是 否 已 成 功 创建 ， 该 命令 用 于 列 
出 附加 数据 库 中 的 所 有 表 。 


sqlite>.tables 
COMPANY DEPARTMENT 


在 这 里 ， 可 以 看 到 COMPANY 表 出 现 两 次 ， 一 个 是 主 数 据 库 的 COMPANY X, — 
个 是 为 testDB.db 创建 的 'test' 别名 的 test. COMPANY 表 。 您 可 以 使 用 SQLite 
.Schema 命令 得 到 表 的 完整 信息 ， 如 下 所 示 : 


sqlite>.schema COMPANY 
CREATE TABLE COMPANY ( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 


SALARY REAL 


SQLite 删除 表 


SQLite 的 DROP TABLE 语句 用 来 删除 表 定 义 及 其 所 有 相关 数据 、 索 引 、 触 发 器 、 
约束 和 该 表 的 权限 规范 。 


> 使 用 此 命令 时 要 特别 注意 ， 因 为 一 旦 一 个 表 被 删除 ， 表 中 所 有 信息 也 将 永远 丢 


DROP TABLE 语句 的 基本 语法 如 下 。 您 可 以 选择 指定 带 有 表 名 的 数据 库 名 称 ， 如 
下 所 示 : 


DROP TABLE database_name.table_name; 


实例 
让 我 们 先 确 认 COMPANY 表 已 经 存在 ， 然 后 我 们 将 其 从 数据 库 中 删除 。 


sqlite>.tables 
COMPANY test .COMPANY 


这 意味 着 COMPANY 表 已 存在 数据 库 中 ， 接 下 来 让 我 们 把 它 从 数据 库 中 删除 ， 如 
下 


sqlite>DROP TABLE COMPANY; 
sqlite> 


现在 ， 如 果 党 试 TABLES 命令 ， 那 么 将 无 法 找到 COMPANY KRT : 


sqlite>.tables 
sqlite> 


显示 结果 为 空 ， 意 味 着 已 经 成 功 从 数据 库 删 除 表 。 


SQLite Insert 话 句 

SQLite 的 INSERT INTO 语句 用 于 向 数据 库 的 某 个 表 中 添加 新 的 数据 行 。 
语法 

INSERT INTO 语句 有 两 种 基本 语法 ， 如 下 所 示 : 


INSERT INTO TABLE_NAME (columni, column2, column3,...columnN)] 
VALUES (value1, value2, value3,...valueN); 


在 这 里 ，column1, column2,...columnN 是 要 插入 数据 的 表 中 的 列 的 名 称 。 


如 果 要 为 表 中 的 所 有 列 添加 值 ， 您 也 可 以 不 需要 在 SQLite 查询 中 指定 列 名 称 。 但 
要 确保 值 的 顺序 与 列 在 表 中 的 顺序 一 致 。SQLite 的 INSERT INTO 语法 如 下 : 


INSERT INTO TABLE_NAME VALUES (valuei, value2, value3,...valueN); 


实例 
假设 您 已 经 在 testDB.db 中 创建 了 COMPANY 表 ， 如 下 所 示 : 


sqlite> CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL 


); 


现在 ， 下 面 的 语句 将 在 COMPANY 表 中 创建 六 个 记录 : 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (1, 'Paul', 32, 'California', 20000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (2, 'Allen', 25, 'Texas', 15000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (3, 'Teddy', 23, 'Norway', 20000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (4, 'Mark', 25, 'Rich-Mond ', 65000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (5, 'David', 27, 'Texas', 85000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (6, 'Kim', 22, 'South-Hall', 45000.00 ); 


您 也 可 以 使 用 第 二 种 语法 在 COMPANY 表 中 创建 一 个 记录 ， 如 下 所 示 : 
INSERT INTO COMPANY VALUES (7, 'James', 24, 'Houston', 10000.00 ); 
| 


上 面 的 所 有 语句 将 在 COMPANY 表 中 创建 下 列 记 录 。 下 一 章 会 教 您 如 何 从 一 个 表 
中 显示 所 有 这 些 记 录 。 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


使 用 一 个 表 来 填充 另 一 个 表 


您 可 以 通过 在 一 个 有 一 组 字段 的 表 上 使 用 select 语句 ， 填 充 数据 到 零 一 个 表 中 。 下 
面 是 语法 : 


INSERT INTO first_table_name [(columni, column2, ... columnN)] 
SELECT columni, column2, ...columnN 
FROM second_table_name 
[WHERE condition]; 


您 白 时 可 以 先 跳 过 上 面 的 语句 ， 可 以 先 学 习 后 面 章节 中 介绍 的 SELECT 和 
WHERE 子 句 。 


SQLite Select 语句 


SQLite 的 SELECT 32 2 AFM SQLite 数据 库 表 中 获取 数据 ， 以 结果 表 的 形式 返 
回 数据 。 这 些 结 果 表 也 被 称 为 结果 集 。 


语法 
SQLite 的 SELECT 语句 的 基本 语法 如 下 : 


SELECT column1，column2，columnN FROM table_name; 


在 这 里 ，column1, column2... 是 表 的 字段 ， 他 们 的 值 即 是 您 要 获取 的 。 如 果 您 想 获 
取 所 有 可 用 的 字段 ， 那 么 可 以 使 用 下 面 的 语法 : 


SELECT * FROM table_name; 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 385000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 使 用 SELECT 语句 获取 并 显示 所 有 这 些 记 录 。 在 这 里 ， 前 三 个 
命令 被 用 来 设置 正确 格式 化 的 输出 。 


sqlite>.header on 
sqlite>.mode column 
sqlite> SELECT * FROM COMPANY; 


最 后 ， 将 得 到 以 下 的 结果 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


如 果 只 想 获 取 COMPANY 表 中 指定 的 字段 ， 则 使 用 下 面 的 查询 : 


sqlite> SELECT ID, NAME, SALARY FROM COMPANY; 


上 面 的 查询 会 产生 以 下 结果 : 
ID NAME SALARY 
1 Paul 20000.0 
2 Allen 15000.0 
3 Teddy 20000.0 
4 Mark 65000.0 
5 David 85000.0 
6 Kim 45000.0 
7 James 10000.0 


设置 输出 列 的 宽度 


有 时 ， 由 于 要 显示 的 列 的 默认 宽度 导致 mode column， 这 种 情况 下 ， 输 出 被 截 
断 。 此 时 ， 您 可 以 使 用 .width num, num.... 命令 设置 显示 列 的 宽度 ， 如 下 所 示 : 


sqlite>.width 10, 20, 10 
sqlite>SELECT * FROM COMPANY; 


上 面 的 .width 命令 设置 第 一 列 的 宽度 为 10， 第 二 列 的 宽度 为 20， 第 三 列 的 宽度 为 
10。 因 此 上 述 SELECT 语句 将 得 到 以 下 结果 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


«| a 








Schema 信息 


sg He a SS erat A 所 以 当 您 进行 带 有 SQLite 的 编程 
您 要 使 用 下 面 的 带 有 sqlite_master 表 的 SELECT 语句 来 列 出 所 有 在 数据 库 
ao : 


sqlite> SELECT tbl_name FROM sqlite_master WHERE type = 'table'; 


假设 在 testDB.db 中 已 经 存在 唯一 的 COMPANY 表 ， 则 将 产生 以 下 结果 : 


tbl_name 


COMPANY 


您 可 以 列 出 关于 COMPANY 表 的 完整 信息 ， 如 下 所 示 : 


sqlite> SELECT sql FROM sqlite master WHERE type = 'table' AND tbl_ 
a ee 
假设 在 testDB.db 中 已 经 存在 唯一 的 COMPANY 表 ， 则 将 产生 以 下 结果 : 





CREATE TABLE COMPANY ( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 


SALARY REAL 


SQLite 运算 符 


SQLite 运算 符 是 什么 ? 

运算 符 是 一 个 保留 字 或 字符 ， 主 要 用 于 SQLite 语句 的 WHERE 子 句 中 执行 操作 ， 
如 比较 和 算术 运算 。 

运算 符 用 于 指定 SQLite 语句 中 的 条 件 ， 并 在 语句 中 连接 多 个 条 件 。 

。 算术 运算 符 

。 上 比较 运算 符 

。 HRA 

。 位 运算 符 


SQLite 算术 运算 符 
假设 变量 a=10， 变 量 b=20， 则 : 
运算 符 描述 实例 
+ 加 法 - 把 运算 符 两 边 的 值 相 加 a + b 将 得 到 30 
减法 - 左 操作 数 减 去 右 操作 数 a-b 将 得 到 -10 
乘法 - 把 运算 符 两 边 的 值 相 乘 a * b 将 得 到 200 
/ 除法 - 左 操作 数 除 以 右 操 作 数 b/a 将 得 到 2 
% 取 模 - 左 操作 数 除 以 右 操 作 数 后 得 到 的 余数 b % a will give 0 
实例 


下 面 是 SQLite 算术 运算 符 的 简单 实例 : 


sqlite> .mode line 
sqlite> select 10 + 20; 
10 + 20 = 30 


sqlite> select 10 - 20; 
10 - 20 = -10 


sqlite> select 10 * 20; 
10 * 20 = 200 


sqlite> select 10 / 5; 
10 /5=2 


sqlite> select 12 % 5; 
12% 5 =2 


SQLite 比较 运算 符 


假设 变量 a=10， 变 量 b=20， 则 : 


<> 


实例 


描述 实例 
检查 两 个 操作 数 的 值 是 否 相等 ， 如 果 相 等 则 条 件 为 ) 不 为 
检查 两 个 操作 数 的 值 是 否 相等 ， 如 果 相 等 则 条 件 为 @ b) 不 为 
直 
检查 两 个 操作 数 的 值 是 否 相等 ， 如 果 不 相等 则 条 件 为 CED 
检查 两 个 操作 数 的 值 是 否 相等 ， 如 果 不 相等 则 条 件 为 fe <> b) 为 
直 
检查 左 操作 数 的 值 是 否 大 于 右 操 作 数 的 值 ， 如 果 是 则 (a > b) 不 为 
条 件 为 真 。 真 。 
检查 左 操作 数 的 值 是 否 小 于 右 操作 数 的 值 ， 如 果 是 则 (a < b) 
条 件 为 真 =o 真 。 
T aaa 大 于 等 于 右 操 作 数 的 值 ， 如 果 (a >= b) 不 为 
是 则 条 件 为 真 真 。 
See aes 小 于 等 于 右 操作 数 的 值 ， 如 果 (== bie 
是 则 条 件 为 真 真 。 
检查 人 不 小 于 右 操 作 数 的 值 ， 如 果 是 (a 1<b) 为 
则 条 件 为 真 假 。 
el aa 不 大 于 右 操作 数 的 值 ， 如 果 是 (al>b) 为 
则 条 件 为 真 真 。 
假设 COMPANY 表 有 以 下 记录 : 
NAME AGE ADDRESS SALARY 
oe Paul 32 California 20000.0 

Allen 25 Texas 15000.0 

Teddy 23 Norway 20000.0 

Mark 25 Rich-Mond 65000 .0 

David 27 Texas 85000.0 

Kim 22 South-Hall 45000.0 

James 24 Houston 10000.0 


下 面 的 实例 演示 了 各 种 SQLite 比较 运算 符 的 用 法 。 


> 在 这 里 ， 我 们 使 用 WHERE 子 句 ， 这 将 会 在 后 边 单 独 的 一 个 章节 中 讲解 ， 但 现在 
您 需要 明白 ，WHERE 子 句 是 用 来 设置 SELECT 语句 的 条 件 语句 。 


下 面 的 SELECT 语句 列 出 了 SALARY 大 于 50,000.00 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE SALARY > 50000; 


ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


下 面 的 SELECT 语句 列 出 了 SALARY 等 于 20,000.00 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE SALARY = 20000; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
3 Teddy 23 Norway 20000.0 


下 面 的 SELECT 语句 列 出 了 SALARY TF 20,000.00 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE SALARY != 20000; 


ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000 .0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 SELECT 语句 列 出 了 SALARY 44 20,000.00 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE SALARY <> 20000; 


ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 SELECT 语句 列 出 了 SALARY 大 于 等 于 65,000.00 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE SALARY >= 65000; 


ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


SQLite #¥ 2 AH 
FE SQLite 中 所 有 的 逻辑 运算 符 列表 。 
运算 符 描述 


AND 


BETWEEN 


EXISTS 


GLOB 
NOT 


OR 


IS NULL 


UNIQUE 


实例 


AND 运算 符 人 允许 在 一 个 SQL 语句 的 WHERE 子 句 中 的 多 个 条 件 
的 存在 。 


BETWEEN 运算 符 用 于 在 给 定 最 小 值 和 最 大 值 范 围 内 的 一 系列 值 
中 搜索 值 。 


EXISTS 运算 符 用 于 在 满足 一 定 条 件 的 指定 表 中 搜索 行 的 存在 。 
IN 运算 符 用 于 把 某 个 值 与 一 系列 指定 列表 的 值 进行 比较 。 


IN 运算 符 的 对 立 面 ， 用 于 把 某 个 值 与 不 在 一 系列 指定 列表 的 值 进 
行 比较 。 
LIKE 运算 符 用 于 把 某 个 值 与 使 用 通配符 运算 符 的 相似 值 进行 比 


较 。 


GLOB 运算 符 用 于 把 某 个 值 与 使 用 通配符 运算 符 的 相似 值 进行 比 
较 。GLOB 与 LIKE 不 同 之 处 在 于 ， 它 是 大 小 写 敏感 的 。 


NOT 运算 符 是 所 用 的 逻辑 运算 符 的 对 立 面 。 比 如 NOT 
EXISTS, NOT BETWEEN、NOT IN， 等 等 。 它 是 否定 运算 符 。 


OR 运算 符 用 于 结合 一 个 SQL 语句 的 WHERE 子 句 中 的 多 个 条 


o 


NULL 运算 符 用 于 把 某 个 值 与 NULL 值 进行 比较 。 
IS 运算 符 与 = 相似 。 

IS NOT 运算 符 与 != 相似 。 

连接 两 个 不 同 的 字符 串 ， 得 到 一 个 新 的 字符 串 。 


ee 运算 符 搜索 指定 表 中 的 每 一 行 ， 确 保 唯 一 性 (无 重 
复 ) 。 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 实例 演示 了 SQLite 逻辑 运算 符 的 用 法 。 


下 面 的 SELECT 语句 列 出 了 AGE 大 于 等 于 25 且 工 资 大 于 等 于 65000.00 的 所 有 
记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE >= 25 AND SALARY >= 65000; 


ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


下 面 的 SELECT 语句 列 出 了 AGE AF SF 25 或 工资 大 于 等 于 65000.00 的 所 有 
记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE >= 25 OR SALARY >= 65000; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


SS | | 


下 面 的 SELECT 语句 列 出 了 AGE 不 为 NULL 的 所 有 记录 ， 结 果 显 示 所 有 的 记录 ， 
意味 着 没有 一 个 记录 的 AGE 等 于 NULL : 


sqlite> SELECT * FROM COMPANY WHERE AGE IS NOT NULL; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000 .0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 SELECT 语句 列 出 了 NAME 以 'Ki' 开始 的 所 有 记录 ，'Ki' 之 后 的 字符 不 做 限 
制 | : 


sqlite> SELECT * FROM COMPANY WHERE NAME LIKE 'Ki%'; 
ID NAME AGE ADDRESS SALARY 


6 Kim 22 South-Hall 45000.0 


下 面 的 SELECT 语句 列 出 了 NAME 以 'Ki' 开始 的 所 有 记录 ，'Ki' 之 后 的 字符 不 做 限 
制 | : 


sqlite> SELECT * FROM COMPANY WHERE NAME GLOB 'Ki*'; 
ID NAME AGE ADDRESS SALARY 


6 Kim 22 South-Hall 45000.0 


下 面 的 SELECT 语句 列 出 了 AGE 的 值 为 25 或 27 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE IN ( 25, 27 ); 


ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


下 面 的 SELECT 语句 列 出 了 AGE 的 值 既 不 是 25 也 不 是 27 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE NOT IN ( 25, 27 ); 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
3 Teddy 23 Norway 20000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


FHA SELECT 语句 列 出 了 AGE 的 值 在 25 与 27 之 间 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE BETWEEN 25 AND 27; 


ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


下 面 的 SELECT 语句 使 用 SQL 子 查询 ， 子 查询 查找 SALARY > 65000 的 带 有 
AGE 字段 的 所 有 记录 ， 后 边 的 WHERE 子 句 与 EXISTS 运算 符 一 起 使 用 ， 列 出 了 
外 查询 中 的 AGE 存在 于 子 查询 返回 的 结果 中 的 所 有 记录 : 


sqlite> SELECT AGE FROM COMPANY 
WHERE EXISTS (SELECT AGE FROM COMPANY WHERE SALARY > 65000 





下 面 的 SELECT 语句 使 用 SQL 子 查询 ， 子 查询 查找 SALARY > 65000 的 带 有 
AGE 字段 的 所 有 记录 ， 后 边 的 WHERE 子 句 与 > 运算 符 一 起 使 用 ， 列 出 了 外 查询 
中 的 AGE 大 于 子 查询 返回 的 结果 中 的 年 龄 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY 
WHERE AGE > (SELECT AGE FROM COMPANY WHERE SALARY > 65000) , 


ID NAME AGE ADDRESS SALARY 

1 Paul 32 California 20000.0 
| 
SQLite 位 运算 符 


位 运算 符 作 用 于 位 ， 并 逐 位 执行 操作 。 真 值 表 & 和 | 如 下 : 


p q 
0 0 
1 0 
1 1 
0 0 


p&q 


假设 如 果 人 A=60， 且 B= 13， 现 在 以 二 进 制 格式 ， 它 们 如 下 所 示 : 


0011 1100 


0000 1101 


= 0000 1100 


= 0011 1101 


= 1100 0011 


下 表 中 列 出 了 SQLite 语言 支持 的 位 运算 符 。 假 设 变 量 A=60， 变 量 B=13， 则 : 


TET 


<< 


>> 


实例 


描 


学 


如 果 同 时 存在 于 两 个 操作 数 中 ， 二 
进 制 AND 运算 符 复制 一 位 到 结果 


o 


如 果 存 在 于 任 一 操作 数 中 ， 二 进 制 
OR 运算 符 复 制 一 位 到 结果 中 。 


二 进 制 补 码 运算 符 是 一 元 运算 符 ， 
具有 "翻转 "位 效应 。 


二 进 制 左 移 运 算 符 。 左 操作 数 的 值 
向 左 移动 右 操作 数 指定 的 位 数 。 


二 进 制 右 移 运 算 符 。 左 操作 数 的 值 
向 右 移 动 右 操作 数 指定 的 位 数 。 


实例 


(A & B) 将 得 到 12， 即 为 0000 
1100 


(A| B) 将 得 到 61， 即 为 0011 1101 
(~A) 将 得 到 -61， 即 为 1100 
0011, 的 补 码 形式 ， 带 符号 的 二 
进 制 数 。 


A << 2 将 得 到 240， 即 为 1111 
0000 


A>> 2 将 得 到 15， 即 为 0000 1111 


下 面 的 实例 演示 了 SQLite 位 运算 符 的 用 法 : 


sqlite> .mode line 
sqlite> select 60 | 13; 
60 | 13 = 61 


sqlite> select 60 & 13; 
60 & 13 = 12 


sqlite> select 60 4 13; 
10 * 20 = 200 


sqlite> select (-~60); 
(~60) = -61 


sqlite> select (60 << 2); 
(60 << 2) = 240 


sqlite> select (60 >> 2); 
(60 >> 2) = 15 


SQLite 表达 陈 


表达 式 是 一 个 或 多 个 值 、 运 算 符 和 计算 值 的 SQL 画 数 的 组 合 。 


SQL 表达 式 与 公式 类 似 ， 都 写 在 查询 语言 中 。 您 还 可 以 使 用 特定 的 数据 集 来 查询 数 
据 库 。 


语法 
假设 SELECT 语句 的 基本 语法 如 下 : 


SELECT columni, column2, columnN 
FROM table_name 
WHERE [CONTION | EXPRESSION]; 


有 不 同类 型 的 SQLite 表达 式 ， 具 体 讲解 如 下 : 


SQLite - 布尔 表达 式 
SQLite 的 布尔 表达 式 在 匹配 单个 值 的 基础 上 获取 数据 。 话 法 如 下 : 


SELECT columni, column2, columnN 
FROM table_name 
WHERE SINGLE VALUE MATCHTING EXPRESSION; 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 实例 演示 了 SQLite 布尔 表达 式 的 用 法 : 


sqlite> SELECT * FROM COMPANY WHERE SALARY = 10000; 


ID NAME AGE ADDRESS SALARY 
-ee ER ye 人 
SQLite - 数值 表达 式 


这 些 表 达 式 用 来 执行 查询 中 的 任何 数学 运算 。 语 法 如 下 : 


SELECT numerical_expression as OPERATION_NAME 
[FROM table name WHERE CONDITION] ; 


ERE, numerical_expression 用 于 数学 表达 了 式 或 任何 公式 。 下 面 的 实例 演示 了 
SQLite 数值 表达 式 的 用 法 : 


sqlite> SELECT (15 + 6) AS ADDITION 
ADDITION = 21 


有 几 个 内 置 的 函数 ， 上 比如 avg()、sum()、count()， 等 等 ， 执 行 被 称 为 对 一 个 表 或 一 
个 特定 的 表 列 的 汇总 数据 计算 。 


sqlite> SELECT COUNT(*) AS "RECORDS" FROM COMPANY; 
RECORDS = 7 


SQLite - 日 期 表达 式 
日 期 表达 式 返 回 当 前 系统 日 期 和 时 间 值 ， 这 些 表 达 式 将 被 用 于 各 种 数据 操作 。 


sqlite> SELECT CURRENT_TIMESTAMP; 
CURRENT_TIMESTAMP = 2013-03-17 10:43:35 


SQLite Where 子 句 


SQLite 的 WHERE 子 句 用 于 指定 从 一 个 表 或 多 个 表 中 获取 数据 的 条 件 。 


如 果 满 足 给 定 的 条 件 ， 即 为 真 (true) 时 ， 则 从 表 中 返回 特定 的 值 。 您 可 以 使 用 
WHERE 子 句 来 过 滤 记 录 ， 只 获取 需要 的 记录 。 


WHERE 子 句 不 仅 可 用 在 SELECT 语句 中 ， 它 也 可 用 在 UPDATE, DELETE 语句 
中 ， 等 等 ， 这 些 我 们 将 在 随后 的 章节 中 学 习 到 。 


语法 
SQLite 的 带 有 WHERE 子 句 的 SELECT 语句 的 基本 语法 如 下 : 


SELECT columni, column2, columnN 
FROM table_name 
WHERE [condition] 


实例 


您 还 可 以 使 用 比较 或 逻辑 运算 符 指 定 条 件 ， 比 如 >、<、=、LIKE、NOT， 等 等 。 假 
设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000 .0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 实例 演示 了 SQLite 逻辑 运算 符 的 用 法 。 下 面 的 SELECT 语句 列 出 了 AGE 
大 于 等 于 25 且 工 资 大 于 等 于 65000.00 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE >= 25 AND SALARY >= 65000; 


ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


4 一 | 








下 面 的 SELECT 语句 列 出 了 AGE 大 于 等 于 25 或 工资 大 于 等 于 65000.00 的 所 有 
记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE >= 25 OR SALARY >= 65000; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000 ,0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
‘| = pc 





下 面 的 SELECT 语句 列 出 了 AGE 不 为 NULL 的 所 有 记录 ， 结 果 显 示 所 有 的 记录 ， 
意味 着 没有 一 个 记录 的 AGE 等 于 NULL : 


sqlite> SELECT * FROM COMPANY WHERE AGE IS NOT NULL; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 SELECT 语句 列 出 了 NAME 以 'Ki' 开始 的 所 有 记录 ，'Ki' 之 后 的 字符 不 做 限 
制 | : 


sqlite> SELECT * FROM COMPANY WHERE NAME LIKE 'Ki%'; 
ID NAME AGE ADDRESS SALARY 


6 Kim 22 South-Hall 45000.0 


下 面 的 SELECT 语句 列 出 了 NAME 以 'Ki' 开始 的 所 有 记录 ，'Ki' 之 后 的 字符 不 做 限 
制 | : 


sqlite> SELECT * FROM COMPANY WHERE NAME GLOB 'Ki*'; 
ID NAME AGE ADDRESS SALARY 


6 Kim 22 South-Hall 45000.0 


FHA SELECT 语句 列 出 了 AGE 的 值 为 25 或 27 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE IN ( 25, 27 ); 


ID 


NAME 


AGE 


ADDRESS 
Texas 
Rich-Mond 
Texas 


SALARY 
15000 .0 
65000 .0 
85000 .0 


下 面 的 SELECT 语句 列 出 了 AGE 的 值 既 不 是 25 也 不 是 27 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE NOT IN ( 25, 27 ); 


NAME 


AGE 


ADDRESS 


California 
Norway 
South-Hall 
Houston 


SALARY 
20000 .0 
20000 .0 
45000.0 
10000 .0 


下 面 的 SELECT 语句 列 出 了 AGE 的 值 在 25 与 27 之 间 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE BETWEEN 25 AND 27; 


ID 


NAME 


AGE 


ADDRESS 


Texas 


Rich-Mond 


Texas 


SALARY 
15000 .0 
65000 .0 
85000 .0 


下 面 的 SELECT 语句 使 用 SQL 子 查询 ， 子 查询 查找 SALARY > 65000 的 带 有 
AGE 字段 的 所 有 记录 ， 后 边 的 WHERE 子 句 与 EXISTS 运算 符 一 起 使 用 ， 列 出 了 
外 查询 中 的 AGE 存在 于 子 查询 返回 的 结果 中 的 所 有 记录 : 


sqlite> SELECT AGE FROM COMPANY 
WHERE EXISTS (SELECT AGE FROM COMPANY WHERE SALARY > 65000: 





下 面 的 SELECT 语句 使 用 SQL 子 查询 ， 子 查询 查找 SALARY > 65000 的 带 有 
AGE 字段 的 所 有 记录 ， 后 边 的 WHERE 子 句 与 > 运算 符 一 起 使 用 ， 列 出 了 外 查询 
中 的 AGE 大 于 子 查 询 返 回 的 结果 中 的 年 龄 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY 
WHERE AGE > (SELECT AGE FROM COMPANY WHERE SALARY > 65000) , 
ID NAME AGE ADDRESS SALARY 


íl Paul 32 California 20000.0 
= | 





SQLite AND/OR 运算 符 


SQLite 的 AND 和 OR 运算 符 用 于 编译 多 个 条 件 来 缩小 在 SQLite 语句 中 所 选 的 数 
据 。 这 两 个 运算 符 被 称 为 连接 运算 符 。 


这 些 运算 符 为 同一 个 SQLite 语句 中 不 同 的 运算 符 之 间 的 多 个 比较 提供 了 可 能 。 
AND 运算 符 

AND 运算 符 允 许 在 一 个 SQL 语句 的 WHERE 子 句 中 的 多 个 条 件 的 存在 。 使 用 
AND 运算 符 时 ， 只 有 当 所 有 条 件 都 为 真 (true) 时 ， 整 个 条 件 为 真 (true) 。 例 


如 ， 只 有 当 condition1 和 condition2 都 为 真 (true) 时 ，[condition1] AND 
[condition2] 为 真 (true) 。 


语法 
带 有 WHERE 子 句 的 AND 运算 符 的 基本 语法 如 下 : 


SELECT columni, column2, columnN 
FROM table_name 
WHERE [condition1] AND [condition2]...AND [conditionN]; 


您 可 以 使 用 AND 运算 符 来 结合 N 个 数量 的 条 件 。SQLite 语句 需要 执行 的 动作 是 ， 
无 论 是 事务 或 查询 ， 所 有 由 AND 分 隔 的 条 件 都 必须 为 真 (TRUE) 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000 .0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


FEN SELECT 语句 列 出 了 AGE 大 于 等 于 25 BT AAFSF 65000.00 的 所 有 
记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE >= 25 AND SALARY >= 65000; 


ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
‘| | 





OR 运算 符 
OR 运算 符 也 用 于 结合 一 个 SQL 语句 的 WHERE 子 句 中 的 多 个 条 件 。 使 用 OR 运 
算 符 时 ， 只 要 当 条 件 中 任何 一 个 为 真 (true) 时 ， 整 个 条 件 为 真 (true) 。 例 如 ， 


只 要 当 condition1 或 condition2 有 一 个 为 真 (true) 时 ，[condition1] OR 
[condition2] 为 真 (true) 。 


语法 
带 有 WHERE 子 句 的 OR 运算 符 的 基本 语法 如 下 : 


SELECT columni, column2, columnN 
FROM table_name 
WHERE [condition1i] OR [condition2]...OR [conditionN] 


您 可 以 使 用 OR 运算 符 来 结合 N 个 数量 的 条 件 。SQLite 语句 需要 执行 的 动作 是 ， 
无 论 是 事务 或 查询 ， 只 要 任何 一 个 由 OR 分 隔 的 条 件 为 真 (TRUE) 即 可 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 

1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000 .0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 的 SELECT 语句 列 出 了 AGE AF SF 25 或 工资 大 于 等 于 65000.00 的 所 有 
记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE >= 25 OR SALARY >= 65000; 


NAME 


AGE 


ADDRESS 
California 
Texas 
Rich-Mond 
Texas 


SALARY 
20000 .0 
15000 .0 
65000 .0 
85000 .0 


m] 





SQLite Update 语句 


SQLite 的 UPDATE 查询 用 于 修改 表 中 已 有 的 记录 。 可 以 使 用 带 有 WHERE 子 句 的 
UPDATE 查询 来 更 新 选 定 行 ， 否 则 所 有 的 行 都 会 被 更 新 。 


语法 
带 有 WHERE 子 句 的 UPDATE 查询 的 基本 语法 如 下 : 


UPDATE table_name 
SET columni = value1, column2 = value2...., columnN = valueN 
WHERE [condition]; 


您 可 以 使 用 AND 或 OR 运算 符 来 结合 N 个 数量 的 条 件 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 它 会 更 新 ID 为 6 的 客户 地 址 : 


sqlite> UPDATE COMPANY SET ADDRESS = 'Texas' WHERE ID = 6; 


现在 ，COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 Texas 45000.0 
7 James 24 Houston 10000.0 


如 果 您 想 修 改 COMPANY 表 中 ADDRESS 和 SALARY 列 的 所 有 值 ， 则 不 需要 使 用 
WHERE 子 句 ，UPDATE 查询 如 下 : 


sqlite> UPDATE COMPANY SET ADDRESS = 'Texas', SALARY = 20000.00; 


mE, COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
alk Paul 32 Texas 20000.0 
2 Allen 25 Texas 20000.0 
3 Teddy 23 Texas 20000.0 
4 Mark 25 Texas 20000.0 
5 David 27 Texas 20000.0 
6 Kim 22 Texas 20000.0 
7 James 24 Texas 20000.0 


SQLite Delete 话 句 


SQLite 的 DELETE 查询 用 于 删除 表 中 已 有 的 记录 。 可 以 使 用 带 有 WHERE 子 句 的 
DELETE 查询 来 删除 选 定 行 ， 否 则 所 有 的 记录 都 会 被 删除 。 


语法 
带 有 WHERE 子 句 的 DELETE 查询 的 基本 语法 如 下 : 


DELETE FROM table name 
WHERE [condition]; 


您 可 以 使 用 AND 或 OR 运算 符 来 结合 N 个 数量 的 条 件 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 它 会 删除 ID 为 7 的 客户 : 


sqlite> DELETE FROM COMPANY WHERE ID = 7; 


m., COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 


如 果 您 想 要 从 COMPANY 表 中 删除 所 有 记录 ， 则 不 需要 使 用 WHERE 4, 
DELETE 查询 如 下 : 


sqlite> DELETE FROM COMPANY; 


现在 ，COMPANY 表 中 没有 任何 的 记录 ， 因 为 所 有 的 记录 已 经 通过 DELETE 语句 
删除 。 


SQLite Like 子 句 


SQLite 的 LIKE 运算 符 是 用 来 匹配 通配符 指定 模式 的 文本 值 。 如 果 搜 索 表 达 式 与 模 
式 表 达 式 匹配 ，LIKE ZB ARES (true) ， 也 就 是 1。 这 里 有 两 个 通配符 与 
LIKE 运算 符 一 起 使 用 : 

e A25 (%) 

。 下 划 线 () 


百 分 号 (%) 代表 雾 个 、 一 个 或 多 个 数字 或 字符 。 下 划 线 C) 代表 一 个 单一 的 数 
字 或 字符 。 这 些 符号 可 以 被 组 合 使 用 。 

语法 

% 和 _ 的 基本 语法 如 下 : 


SELECT FROM table name 
WHERE column LIKE 'XXXxX%' 


or 


SELECT FROM table_name 
WHERE column LIKE '%XXXX%' 


or 


SELECT FROM table name 
WHERE column LIKE 'XXXX_' 


or 


SELECT FROM table_name 
WHERE column LIKE ' XXXX' 


or 
SELECT FROM table_name 
WHERE column LIKE ' XXXX 


您 可 以 使 用 AND 或 OR 运算 符 来 结合 N 个 数量 的 条 件 。 在 这 里 ，XXXX 可 以 是 任 
何 数字 或 字符 串 值 。 


实例 


下 面 一 些 实例 演示 了 带 有 '%' 和 | 运算 符 的 LIKE 子 名 不同 的 地 方 : 


语句 


WHERE SALARY LIKE 
'200%' 


WHERE SALARY LIKE 
'%.200%' 


WHERE SALARY LIKE 
'_00%' 


WHERE SALARY LIKE 
D % 


WHERE SALARY LIKE '%2' 


WHERE SALARY LIKE 
" 2%3' 


WHERE SALARY LIKE 
2—3 


描述 


查找 以 200 开头 的 任意 值 


查找 任意 位 置 包含 200 的 任意 值 


查找 第 二 位 和 第 三 位 为 00 的 任意 值 


查找 以 2 开头 ， 且 长 度 至 少 为 3 个 字符 的 任意 
值 


查找 以 2 结尾 的 任意 值 
查找 第 二 位 为 2， 且 以 3 结尾 的 任意 值 


查找 长 度 为 5 位 数 ， 且 以 2 开头 以 3 结尾 的 
任意 值 


让 我 们 举 一 个 实际 的 例子 ， 假 设 COMPANY 表 有 以 下 记录 : 


ADDRESS SALARY 
California 20000.0 
Texas 15000.0 
Norway 20000.0 
Rich-Mond 65000.0 
Texas 85000.0 
South-Hall 45000.0 
Houston 10000.0 


下 面 是 一 个 实例 ， 它 显示 COMPANY 表 中 AGE 以 2 开头 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE LIKE '2%'; 


这 将 产生 以 下 结果 : 


ID NAME AGE ADDRESS SALARY 


2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 它 显示 COMPANY 表 中 ADDRESS 文本 里 包含 一 个 连 字符 (-) 
的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE ADDRESS LIKE '%-%'; 


这 将 产生 以 下 结果 : 
ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 


6 Kim 22 South-Hall 45000.0 


SQLite Glob 子 句 


SQLite 的 GLOB 运算 符 是 用 来 匹配 通配符 指定 模式 的 文本 值 。 如 果 搜 索 表 达 式 与 
模式 表达 式 匹 配 ，GLOB 运算 符 将 返回 真 (true) ， 也 就 是 1。 与 LIKE 运算 符 不 同 
的 是 ，GLOB 是 大 小 写 敏 感 的 ， 对 于 下 面 的 通配符 ， 它 遵循 UNIX 的 语法 。 

e #5 (*) 

e 问号 (?) 
BS (*) 代表 需 个 、 一 个 或 多 个 数字 或 字符 。 问 号 (?) 代表 一 个 单一 的 数字 或 字 
符 。 这 些 符号 可 以 被 组 合 使 用 。 
语法 

e 和 ?的 基本 语法 如 下 : 


SELECT FROM table_name 
WHERE column GLOB 'XXXxX*' 


or 


SELECT FROM table_name 
WHERE column GLOB '*XXxXxX*' 


or 


SELECT FROM table_name 
WHERE column GLOB 'XXXxX?' 


or 


SELECT FROM table_name 
WHERE column GLOB '?XXXX' 


or 


SELECT FROM table_name 
WHERE column GLOB '?XXXxX?' 


or 
SELECT FROM table_name 
WHERE column GLOB '????' 


您 可 以 使 用 AND 或 OR 运算 符 来 结合 N 个 数量 的 条 件 。 在 这 里 ，XXXX 可 以 是 任 
何 数字 或 字符 串 值 。 


实例 


下 面 一 些 实例 演示 了 A'M '?' 运算 符 的 GLOB 子 句 不 同 的 地 方 : 


语句 


WHERE SALARY GLOB 


'200*' 


WHERE SALARY GLOB 


#2005 


WHERE SALARY GLOB '? 


00*' 


WHERE SALARY GLOB 


2 


WHERE SALARY GLOB ”2 
WHERE SALARY GLOB '? 


259] 


WHERE SALARY GLOB 


'22??3' 


描述 


查找 以 200 开头 的 任意 值 


查找 任意 位 置 包含 200 的 任意 值 


查找 第 二 位 和 第 三 位 为 00 的 任意 值 


查找 以 2 开头 ， 且 长 度 至 少 为 3 个 字符 的 任意 
值 


查找 以 2 结尾 的 任意 值 
查找 第 二 位 为 2， 且 以 3 结尾 的 任意 值 


查找 长 度 为 5 位 数 ， 且 以 2 开头 以 3 结尾 的 
任意 值 


让 我 们 举 一 个 实际 的 例子 ， 假 设 COMPANY 表 有 以 下 记录 : 


ADDRESS SALARY 
California 20000.0 
Texas 15000.0 
Norway 20000.0 
Rich-Mond 65000.0 
Texas 85000.0 
South-Hall 45000.0 
Houston 10000. 0 


下 面 是 一 个 实例 ， 它 显示 COMPANY 表 中 AGE 以 2 开头 的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE AGE GLOB '2*'; 


这 将 产生 以 下 结果 : 


ID NAME AGE ADDRESS SALARY 


2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 它 显示 COMPANY 表 中 ADDRESS 文本 里 包含 一 个 连 字符 (-) 
的 所 有 记录 : 


sqlite> SELECT * FROM COMPANY WHERE ADDRESS GLOB '*-*'; 


这 将 产生 以 下 结果 : 
ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 


6 Kim 22 South-Hall 45000.0 


SQLite Limit 子 句 

SQLite 的 LIMIT 子 句 用 于 限制 由 SELECT 语句 返回 的 数据 数量 。 
语法 

带 有 LIMIT 子 句 的 SELECT 语句 的 基本 语法 如 下 : 


SELECT columni, column2, columnN 
FROM table_name 
LIMIT [no of rows] 


下 面 是 LIMIT #4)4 OFFSET 子 句 一 起 使 用 时 的 语法 : 


SELECT columni, column2, columnN 
FROM table_name 
LIMIT [no of rows] OFFSET [row num] 


SQLite 引擎 将 返回 从 下 一 行 开始 直到 给 定 的 OFFSET 为 止 的 所 有 行 ， 如 下 面 的 最 
后 一 个 实例 所 示 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 它 限制 了 您 想 要 从 表 中 提取 的 行 数 : 


sqlite> SELECT * FROM COMPANY LIMIT 6; 


这 将 产生 以 下 结果 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 


但 是 ， 在 某 些 情况 下 ， 可 能 需要 从 一 个 特定 的 偏 移 开 始 提取 记录 。 下 面 是 一 个 实 
例 ， 从 第 三 位 开始 提取 3 个 记录 : 


sqlite> SELECT * FROM COMPANY LIMIT 3 OFFSET 2; 


这 将 产生 以 下 结果 : 
ID NAME AGE ADDRESS SALARY 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 


5 David 27 Texas 85000.0 


SQLite Order By 

SQLite 的 ORDER BY 子 句 是 用 来 基于 一 个 或 多 个 列 按 升序 或 降序 顺序 排列 数据 。 
语法 

ORDER BY 子 名 的 基本 语法 如 下 : 


SELECT column-list 

FROM table_name 

[WHERE condition] 

[ORDER BY columni, column2, .. columnN] [ASC | DESC]; 


您 可 以 在 ORDER BY 子 句 中 使 用 多 个 列 。 确 保 您 使 用 的 排序 列 在 列 清单 中 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 是 一 个 实例 ， 它 会 将 结果 按 SALARY 降序 排序 : 


sqlite> SELECT * FROM COMPANY ORDER BY SALARY ASC; 


这 将 产生 以 下 结果 : 


ID NAME AGE ADDRESS SALARY 


7 James 24 Houston 10000.0 
2 Allen 25 Texas 15000.0 
1 Paul 32 California 20000.0 
3 Teddy 23 Norway 20000.0 
6 Kim 22 South-Hall 45000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


下 面 是 一 个 实例 ， 它 会 将 结果 按 NAME 和 SALARY 降序 排序 : 


sqlite> SELECT * FROM COMPANY ORDER BY NAME, SALARY ASC; 


ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000.0 
5 David 27 Texas 85000.0 
7 James 24 Houston 10000.0 
6 Kim 22 South-Hall 45000.0 
4 Mark 25 Rich-Mond 65000.0 
1 Paul 32 California 20000.0 
3 Teddy 23 Norway 20000.0 


下 面 是 一 个 实例 ， 它 会 将 结果 按 NAME 降序 排序 : 


sqlite> SELECT * FROM COMPANY ORDER BY NAME DESC; 


ID NAME AGE ADDRESS SALARY 
3 Teddy 23 Norway 20000.0 
1 Paul 32 California 20000.0 
4 Mark 25 Rich-Mond 65000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000 ,0 
5 David 27 Texas 85000.0 
2 Allen 25 Texas 15000.0 


SQLite Group By 


SQLite 的 GROUP BY 子 句 用 于 与 SELECT 语句 一 起 使 用 ， 来 对 相同 的 数据 进行 
分 组 。 


在 SELECT 语句 中 ，GROUP BY 子 句 放 在 WHERE 子 句 之 后 ， 放 在 ORDER BY 
子 句 之 前 。 


语法 


下 面 给 出 了 GROUP BY 子 句 的 基本 语法 。GROUP BY 子 句 必须 放 在 WHERE 子 
名 中 的 条 件 之 后 ， 必 须 放 在 ORDER BY 子 句 之 前 。 


SELECT column-list 

FROM table_name 

WHERE [ conditions | 

GROUP BY columni, column2....columnN 
ORDER BY columni, column2....columnN 


您 可 以 在 GROUP BY 子 句 中 使 用 多 个 列 。 确 保 您 使 用 的 分 组 列 在 列 清单 中 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000 .0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


如 果 您 想 了 解 每 个 客户 的 工资 总 额 ， 则 可 使 用 GROUP BY 查询 ， 如 下 所 示 : 


sqlite> SELECT NAME, SUM(SALARY) FROM COMPANY GROUP BY NAME; 


这 将 产生 以 下 结果 : 


NAME SUM(SALARY ) 


Allen 15000 .0 
David 85000 .0 
James 10000 .0 
Kim 45000.0 
Mark 65000 .0 
Paul 20000.0 
Teddy 20000.0 


现在 ， 让 我 们 使 用 下 面 的 INSERT 语句 在 COMPANY 表 中 另外 创建 三 个 记录 : 


INSERT INTO COMPANY VALUES (8, 'Paul', 24, 'Houston', 20000.00 ); 
INSERT INTO COMPANY VALUES (9, '‘James', 44, 'Norway', 5000.00 ); 
INSERT INTO COMPANY VALUES (10, 'James', 45, 'Texas', 5000.00 ); 


[ER 
现在 ， 我 们 的 表 具 有 重复 名 称 的 记录 ， 如 下 所 示 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 
8 Paul 24 Houston 20000 .0 
9 James 44 Norway 5000.0 
10 James 45 Texas 5000.0 


让 我 们 用 同样 的 GROUP BY 语句 来 对 所 有 记录 按 NAME 列 进 行 分 组 ， 如 下 所 示 : 


sqlite> SELECT NAME, SUM(SALARY) FROM COMPANY GROUP BY NAME ORDER 上 





SUM(SALARY) 


让 我 们 把 ORDER BY 子 句 与 GROUP BY 子 句 一 起 使 用 ， 如 下 所 示 : 


sqlite> SELECT NAME, SUM(SALARY) 
FROM COMPANY GROUP BY NAME ORDER BY NAME DESC; 


SUM (SALARY ) 


SQLite Having 子 句 


HAVING 子 句 允许 指定 条 件 来 过 滤 将 出 现在 最 终结 果 中 的 分 组 结果 。 


WHERE 子 句 在 所 选 列 上 设置 条 件 ， 而 HAVING 子 句 则 在 由 GROUP BY 子 句 创建 
的 分 组 上 设置 条 件 。 


语法 
下 面 是 HAVING 子 句 在 SELECT 查询 中 的 位 置 : 


SELECT 
FROM 
WHERE 
GROUP BY 
HAVING 
ORDER BY 


在 一 个 查询 中 ，HAVING 子 句 必 须 放 在 GROUP BY 子 名 之后， 必须 放 在 ORDER 
BY 子 句 之 前 。 下 面 是 包含 HAVING 子 句 的 SELECT 语句 的 语法 : 


SELECT column1, column2 
FROM table1, table2 

WHERE [ conditions ] 
GROUP BY columni, column2 
HAVING [ conditions ] 
ORDER BY columni, column2 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 
8 Paul 24 Houston 20000.0 
9 James 44 Norway 5000.0 

10 James 45 Texas 5000.0 


下 面 是 一 个 实例 ， 它 将 显示 名 称 计 数 小 于 2 的 所 有 记录 : 


sqlite > SELECT * FROM COMPANY GROUP BY name HAVING count(name) < : 





ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000 
5 David 27 Texas 85000 
6 Kim 22 South-Hall 45000 
4 Mark 25 Rich-Mond 65000 
3 Teddy 23 Norway 20000 


下 面 是 一 个 实例 ， 它 将 显示 名 称 计 数 大 于 2 的 所 有 记录 : 


sqlite > SELECT * FROM COMPANY GROUP BY name HAVING count(name) > : 
<] == $ 
这 将 产生 以 下 结果 : 





ID NAME AGE ADDRESS SALARY 


SQLite Distinct 关键 字 
SQLite 的 DISTINCT 关键 字 与 SELECT 语句 一 起 使 用 ， 来 消除 所 有 重复 的 记录 ， 
并 只 获取 唯一 一 次 记录 。 


有 可 能 出 现 一 种 情况 ， 在 一 个 表 中 有 多 个 重复 的 记录 。 当 提取 这 样 的 记录 时 ， 
DISTINCT 关键 字 就 显得 特别 有 意义 ， 它 只 获取 唯一 一 次 记录 ， 而 不 是 获取 重复 记 
录 。 


语法 
用 于 消除 重复 记录 的 DISTINCT 关键 字 的 基本 语法 如 下 : 


SELECT DISTINCT columni, column2,..... columnN 
FROM table_name 
WHERE [condition] 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 
8 Paul 24 Houston 20000.0 
9 James 44 Norway 5000.0 
10 James 45 Texas 5000.0 


首先 ， 让 我 们 来 看 看 下 面 的 SELECT 查询 ， 它 将 返回 重复 的 工资 记录 : 


sqlite> SELECT name FROM COMPANY; 


这 将 产生 以 下 结果 : 


现在 ， 让 我 们 在 上 述 的 SELECT 查询 中 使 用 DISTINCT 关键 字 : 


sqlite> SELECT DISTINCT name FROM COMPANY; 


这 将 产生 以 下 结果 ， 没 有 任何 重复 的 条 目 : 


SQLite 基础 


SQLite PRAGMA 


SQLite 的 PRAGMA 命令 是 一 个 特殊 的 命令 ， 可 以 用 在 SQLite 环境 内 控制 各 种 环 
境 变量 和 状态 标志 。 一 个 PRAGMA 值 可 以 被 读 取 ， 也 可 以 根据 需求 进行 设置 。 


语法 
要 查询 当前 的 PRAGMA 值 ， 只 需要 提供 该 pragma 的 名 字 : 
PRAGMA pragma_name; 


要 为 PRAGMA 设置 一 个 新 的 值 ， 话 法 如 下 : 


PRAGMA pragma_name = value; 


设置 模式 ， 可 以 是 名 称 或 等 值 的 整数 ， 但 返回 的 值 将 始终 是 一 个 整数 。 


auto_vacuum Pragma 
auto_vacuum Pragma 获取 或 设置 auto-vacuum 模式 。 语 法 如 下 : 


PRAGMA [database. ]auto_vacuum; 
PRAGMA [database. ]auto_vacuum = mode; 


Er, mode 可 以 是 以 下 任何 一 种 : 


Pragma 值 描述 

0 或 NONE 禁用 Auto-vacuum。 这 是 默认 模式 ， 意 味 着 数据 库 文件 尺寸 
K 大 小 不 会 缩小 ， 除 非 手动 使 用 VACUUM 命 兮 。 

1 或 FULL 启用 Auto-vacuum， 是 全 自动 的 。 在 该 模式 下 ， 人 允许 数据 库 


文件 随 着 数据 从 数据 库 移 除 而 缩小 。 


2 或 启用 Auto-vacuum， 但 是 必须 手动 激活 。 在 该 模式 下 ， 引 用 
INCREMENTAL 数据 被 维持 ， 免 费 页 面 只 放 在 免费 列表 中 。 这 些 页 面 可 在 任 
何 时 候 使 用 incremental_vacuum pragma 进行 覆盖 。 


cache_size Pragma 


cache_size Pragma 可 获取 或 暂时 设置 在 内 存 中 页 面 缓存 的 最 大 尺寸 。 语 法 如 下 : 


PRAGMA [database. ]cache_size; 
PRAGMA [database.]cache_size = pages; 


pages 值 表示 在 缓存 中 的 页 面 数 。 内 置 页 面 缓存 的 默认 大 小 为 2,000 页 ， 最 小 尺寸 
为 10 页 。 


case_sensitive_like Pragma 


case_sensitive_like Peay 控制 内 置 的 LIKE 表达 式 的 大 小 写 敏感 度 。 默 认 情 况 
下 ， 该 Pragma 为 false， 这 意味 着 ， 内 置 的 LIKE 操作 符 忽 略 字 母 的 大 小 写 。 语 法 
如 下 : 


PRAGMA case_sensitive_like = [true|false]; 
目前 没有 办 法 查询 该 Pragma 的 当前 状态 。 


count_changes Pragma 


count_changes Pragma 获取 或 设置 数据 操作 语句 的 返回 值 ， 如 INSERT, 
UPDATE 和 DELETE。 语 法 如 下 : 


PRAGMA count_changes; 
PRAGMA count_changes = [true|false]; 


默认 情况 下 ， 该 Pragma 为 false， 这 些 语句 不 返回 任何 东西 。 如 果 设 置 a o 
每 个 所 提 到 的 语句 将 返回 一 个 单行 单列 的 表 ， 由 一 个 单一 的 整数 值 组 成 ， 该 整数 表 
示 操 作 影 响 的 行 。 


database_list Pragma 
database_list Pragma 将 用 于 列 出 了 所 有 的 数据 库 连 接 。 语 法 如 下 : 


PRAGMA database_list; 


该 Pragma 将 返回 一 个 单行 三 列 的 表格 ， 每 当 打开 或 附加 数据 库 时 ， 会 给 出 数据 库 
中 的 序列 号 ， 它 的 名 称 和 相关 的 文件 。 


encoding Pragma 
encoding Pragma 控制 字符 串 如 何 编码 及 存储 在 数据 库 文件 中 。 语 法 如 下 : 


PRAGMA encoding; 
PRAGMA encoding = format; 


格式 值 可 以 是 UTF-8、UTF-16le 或 UTF-16be 之 一 。 


freelist count Pragma 


freelist_count Pragma 返回 一 个 整数 ， 表 示 当 前 被 标记 为 免费 和 可 用 的 数据 库 页 
数 。 语 法 如 下 : 


PRAGMA [database.]freelist_count; 


格式 值 可 以 是 UTF-8、UTF-16le 或 UTF-16be 之 一 。 


index_info Pragma 
index_info Pragma 返回 关于 数据 库 索 引 的 信息 。 语 法 如 下 : 


PRAGMA [database. ]index_info( index_name ); 


结果 集 将 为 每 个 包含 在 给 出 列 序 列 的 索引 、 表 格 内 的 列 素 引 、 列 名 称 的 列 显示 一 
行 。 


index_list Pragma 
index_list Pragma 列 出 所 有 和 与 表 相 关联 的 索引 。 语 法 如 下 : 


PRAGMA [database. ]index_list( table_name ); 


结果 集 将 为 每 个 给 出 列 序 列 的 索引 、 索 引 名 称 、 表 示 索 引 是 否 唯一 的 标识 显示 一 
行 。 


Jjournal _ mode Pragma 


journal _mode Pragma 获取 或 设置 控制 日 志文 件 如 何 存储 和 义理 的 日 志 模 式 。 语 
法 如 下 :: 


PRAGMA journal_mode; 

PRAGMA journal_mode = mode; 

PRAGMA database. journal_mode; 

PRAGMA database. journal_mode = mode; 


这 里 支持 五 种 日 志 模式 : 
Pragma 值 描述 
DELETE 默认 模式 。 在 该 模式 下 ， 在 事务 结束 时 ， 日 志文 件 将 被 删除 
TRUNCATE ” 日志 文件 被 阶段 为 规 字 节 长 度 。 


PERSIST 日 志文 件 被 留 在 原 地 ， 但 头 部 被 重 写 ， 表 明日 志 不 再 有 效 。 
MEMORY 日 志 记 录 保 留 在 内 存 中 ， 而 不 是 磁盘 上 。 
OFF 不 保留 加 任何 日 志 记 录 。 


max_page_count Pragma 
max_page_count Pragma 为 数据 库 获取 或 设置 允许 的 最 大 页 数 。 语 法 如 下 : 


PRAGMA [database. ]max_page_count; 
PRAGMA [database. ]max_page_count = max_page; 


默认 值 是 1,073,741,823， 这 是 一 个 千 兆 的 页 面 ， 即 如 果 默 认 1 KB 的 页 面 大 小 ， 那 
么 数据 库 中 增长 起 来 的 一 个 兆 字 节 。 


page_count Pragma 
page_count Pragma 返回 当前 数据 库 中 的 网 页 数量 。 语 法 如 下 : 
PRAGMA [database. ]page_count; 


数据 库 文件 的 大 小 应 该 是 page_count * page_size. 


page_size Pragma 


page_size Pragma 获取 或 设置 数据 库 页 面 的 大 小 。 语 法 如 下 : 


PRAGMA [database. ]page_size; 
PRAGMA [database. ]page_size = bytes; 


默认 情况 下 ， 人 允许 的 尺寸 是 512、1024、2048、4096、8192、16384、32768 = 
节 。 改 变现 有 数据 库 页面 大 小 的 唯一 方法 就 是 设置 页 面 大 小 ， 然 后 立即 VACUUM 
该 数据 库 。 


parser trace Pragma 
parser_trace Pragma 随 着 它 解 析 SQL 命令 来 控制 打印 的 调试 状态 ， 语 法 如 下 : 


PRAGMA parser_trace = [true|false]; 


默认 情况 下 ， 它 被 设置 为 false， 但 设置 为 true 时 则 启用 ， 此 时 SQL 解析 器 会 随 着 
它 解析 SQL 命令 来 打印 出 它 的 状态 。 


recursive_triggers Pragma 


recursive_triggers Pragma 获取 或 设置 递归 触发 器 功能 。 如 果 未 启用 递 轨 触发 
器 ， 一 个 触发 动作 将 不 会 触发 另 一 个 触发 。 语 法 如 下 : 


PRAGMA recursive_triggers; 
PRAGMA recursive_triggers = [true|false]; 
schema_version Pragma 


schema_version Pragma 获取 或 设置 存储 在 数据 库 头 中 的 的 架构 版 本 值 。 语 法 如 
F: 


PRAGMA [database.]schema_version; 
PRAGMA [database.]schema_version = number; 


这 是 一 个 32 位 有 符号 整数 值 ， 用 来 跟踪 架构 的 变化 。 每 当 一 个 架构 改变 命令 执 
(比如 CREATE... 或 DROP...) 时 ， 这 个 值 会 递增 。 


secure_delete Pragma 


secure_delete Pragma 用 来 控制 内 容 是 如 何 从 数据 库 中 删除 。 话 法 如 下 : 


PRAGMA secure_delete; 

PRAGMA secure_delete = [true|false]; 

PRAGMA database.secure_delete; 

PRAGMA database.secure_delete = [true|false]; 


安全 删除 标志 的 默认 值 通常 是 关闭 的 ， 但 是 这 是 可 以 通过 
SQLITE_SECURE_DELETE 构建 选项 来 改变 的 。 


sql_ trace Pragma 
sql_trace Pragma 用 于 把 SQL 跟踪 结果 转 储 到 屏幕 上 。 语 法 如 下 : 


PRAGMA sql_trace; 
PRAGMA sql_trace = [true|false]; 


SQLite 必须 通过 SQLITE_DEBUG 指令 来 编译 要 引用 的 该 Pragma。 


synchronous Pragma 


synchronous Pragma 获取 或 设置 当前 磁 旬 的 同步 模式 ， 该 模式 控制 积极 的 
SQLite 如 何 将 数据 写 入 物理 存储 。 语 法 如 下 : 


PRAGMA [database. ]synchronous; 
PRAGMA [database.]synchronous = mode; 


SQLite 支持 下 列 同步 模式 : 


Pragma 值 描述 
0 或 OFF 不 进行 同步 。 
1 或 NORMAL 在 关键 的 磁盘 操作 的 每 个 序列 后 同步 。 
2 或 FULL 在 每 个 关键 的 磁盘 操作 后 同步 。 


temp_store Pragma 
temp_store Pragma 获取 或 设置 临时 数据 库 文 件 所 使 用 的 存储 模式 。 话 法 如 下 : 


PRAGMA temp_store, 
PRAGMA temp_store = mode; 


SQLite 支持 下 列 存储 模式 : 


Pragma 值 描述 
0 或 DEFAULT 默认 使 用 编译 时 的 模式 。 通 常 是 FILE。 
1 或 FILE 使 用 基于 文件 的 存储 。 
2 或 MEMORY 使 用 基于 内 存 的 存储 。 


temp_store_directory Pragma 


temp_store_directory Pragma 获取 或 设置 用 于 临时 数据 库 文件 的 位 置 。 语 法 如 
下 : 


PRAGMA temp_store_directory; 
PRAGMA temp_store_directory = 'directory_path'; 


user_version Pragma 


user_version Pragma 获取 或 设置 存储 在 数据 库 头 的 用 户 自 定义 的 版 本 值 。 语 法 如 
下 : 


PRAGMA [database. ]user_version; 
PRAGMA [database. ]user_version = number; 


这 是 一 个 32 位 的 有 符号 整数 值 ， 可 以 由 开发 人 员 设 置 ， 用 于 版 本 跟踪 的 目的 。 


writable_schema Pragma 
writable_schema Pragma 获取 或 设置 是 否 能 够 修改 系统 表 。 语 法 如 下 : 


PRAGMA writable_schema; 
PRAGMA writable schema = [true|false]; 


如 果 设 置 了 该 Pragma, WRLA sqlite_ 开 始 ， 可 以 创建 和 修改 ， 包 括 
sqlite_master 表 。 使 用 该 Pragma 时 要 注意 ， 因 为 它 可 能 导致 整个 数据 库 损 坏 。 


SQLite 约束 
约束 是 在 表 的 数据 列 上 强制 执行 的 规则 。 这 些 是 用 来 限制 可 以 插入 到 表 中 的 数据 类 
型 。 这 确保 了 数据 库 中 数据 的 准确 性 和 可 靠 性 。 
约束 可 以 是 列 级 或 表 级 。 列 级 约束 仅 适用 于 列 ， 表 级 约束 被 应 用 到 整个 表 。 
以 下 是 在 SQLite 中 常用 的 约束 。 
e NOT NULL 约束 : 确保 某 列 不 能 有 NULL 值 。 
e DEFAULT 约束 : 当 某 列 没有 指定 值 时 ， 为 该 列 提供 默认 值 。 
e UNIQUE 约束 : 确保 某 列 中 的 所 有 值 是 不 同 的 。 
e PRIMARY Key 约束 : 唯一 标识 数据 库 表 中 的 各 行 /记录 。 
e CHECK 约束 : CHECK 约束 确保 某 列 中 的 所 有 值 满足 一 定 条 件 。 


NOT NULL 约束 


默认 情况 下 ， 列 可 以 保存 NULL 值 。 如 果 您 不 想 某 列 有 NULL 值 ， 那 么 需要 在 该 列 
上 定义 此 约束 ， 指 定 在 该 列 上 不 允许 NULL 值 。 


NULL 与 没有 数据 是 不 一 样 的 ， 它 代表 着 未 知 的 数据 。 
实例 


例如 ， 下 面 的 SQLite 语句 创建 一 个 新 的 表 COMPANY， 并 增加 了 五 列 ， 其 中 ID, 
NAME 和 AGE 三 列 指定 不 接受 NULL 值 : 


CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL 


); 


DEFAULT 约束 


DEFAULT 约束 在 INSERT INTO 语句 没有 提供 一 个 特定 的 值 时 ， 为 列 提供 一 个 默 
认 值 。 


实例 
例如 ， 下 面 的 SQLite 语句 创建 一 个 新 的 表 COMPANY， 并 增加 了 五 列 。 在 这 里 ， 


SALARY 列 默认 设置 为 5000.00。 所 以 当 INSERT INTO 语句 没有 为 该 列 提 供 值 
时 ， 该 列 将 被 设置 为 5000.00。 


CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL, 

NAME TEXT NOT NULL, 

AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL DEFAULT 50000.00 


); 


UNIQUE 约束 


UNIQUE 约束 防止 在 一 个 特定 的 列 存在 两 个 记录 县 有 相同 的 值 。 在 COMPANY R 
中 ， 例 如 ， 您 可 能 要 防止 两 个 或 两 个 以 上 的 人 具有 相同 的 年 龄 。 


实例 


例如 ， 下 面 的 SQLite 语句 创建 一 个 新 的 表 COMPANY， 并 增加 了 五 列 。 在 这 里 ， 
AGE 列 设置 为 UNIQUE， 所 以 不 能 有 两 个 相同 年 龄 的 记录 : 


CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL， 
NAME TEXT NOT NULL， 

AGE INT NOT NULL UNIQUE, 
ADDRESS CHAR(50), 

SALARY REAL DEFAULT 50000.00 


); 


PRIMARY KEY 4 


PRIMARY KEY 约束 唯一 标识 数据 库 表 中 的 每 个 记录 。 在 一 个 表 中 可 以 有 多 个 
UNIQUE 列 ， 但 只 能 有 一 个 主键 。 在 设计 数据 库 表 时 ， 主 键 是 很 重要 的 。 主 键 是 唯 
一 的 ID。 


我 们 使 用 主键 来 引用 表 中 的 行 。 可 通过 把 主键 设置 为 其 他 表 的 外 键 ， 来 创建 表 之 间 
的 关系 。 由 于 "长 期 存在 编码 监督 "， 在 SQLite 中 ， 主 键 可 以 是 NULL， 这 是 与 其 他 
数据 库 不 同 的 地 方 。 


主键 是 表 中 的 一 个 字段 ， 唯 一 标识 数据 库 表 中 的 各 行 /记录 。 主 键 必须 包含 唯一 值 。 
主键 列 不 能 有 NULL 值 。 


一 个 表 只 能 有 一 个 主键 ， 它 可 以 由 一 个 或 多 个 字段 组 成 。 当 多 个 字段 作为 主键 ， 它 
们 被 称 为 复合 键 。 


ee a 
目 同 的 值 。 


实例 
已 经 看 到 了 我 们 创建 以 ID 作为 主键 的 COMAPNY 表 的 各 种 实例 : 


CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL， 
NAME TEXT NOT NULL, 
AGE INT NOT NULL， 
ADDRESS CHAR(50), 

SALARY REAL 


); 


CHECK 约束 


CHECK 约束 启用 输入 一 条 记录 要 检查 值 的 条 件 。 如 果 条 件 值 为 false， 则 记录 违反 
了 约束 ， 且 不 能 输入 到 表 。 


实例 


例如 ， 下 面 的 SQLite 创建 一 个 新 的 表 COMPANY， 并 增加 了 五 列 。 在 这 里 ， 我 们 
为 SALARY 列 添加 CHECK， 所 以 工资 不 能 为 霉 : 


CREATE TABLE COMPANY3( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 

AGE INT NOT NULL, 

ADDRESS CHAR(50), 

SALARY REAL CHECK(SALARY > 0) 


); 


删除 约束 


SQLite 支持 ALTER TABLE 的 有 限 子 集 。 在 SQLite 中 ，ALTER TABLE 命令 允许 
用 户 重 命名 表 ， 或 向 现 有 表 添 加 一 个 新 的 列 。 重 命名 列 ， 删 除 一 列 ， 或 从 一 个 表 中 
添加 或 删除 约束 都 是 不 可 能 的 。 


SQLite Joins 
SQLite 的 Joins 子 句 用 于 结合 两 个 或 多 个 数据 库 中 表 的 记录 。JOIN 是 一 种 通过 共 
同 值 来 结合 两 个 表 中 字段 的 手段 。 
SQL 定义 了 三 种 主要 类 型 的 连接 : 
e 交叉 连接 - CROSS JOIN 
e 内 连接 - INNER JOIN 
e 外 连接 - OUTER JOIN 


在 我 们 继续 之 前 ， 让 我 们 假设 有 两 个 表 COMPANY 和 DEPARTMENT。 我 们 已 经 
看 到 了 用 来 填充 COMPANY 表 的 INSERT 语句 。 现 在 让 我 们 假设 COMPANY 表 的 
记录 列表 如 下 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


另 一 个 表 是 DEPARTMENT， 定 义 如 下 : 


CREATE TABLE DEPARTMENT( 


ID INT PRIMARY KEY NOT NULL, 
DEPT CHAR(50) NOT NULL, 
EMP_ID INT NOT NULL 


); 


下 面 是 填充 DEPARTMENT 表 的 INSERT 语句 : 


INSERT INTO DEPARTMENT (ID, DEPT, EMP_ID) 
VALUES (1, ‘IT Billing’, 1 ); 


INSERT INTO DEPARTMENT (ID, DEPT, EMP_ID) 
VALUES (2, 'Engineering', 2 ); 


INSERT INTO DEPARTMENT (ID, DEPT, EMP_ID) 
VALUES (3, 'Finance', 7 ); 


最 后 ， 我 们 在 DEPARTMENT 表 中 有 下 列 的 记录 列表 : 


ID DEPT EMP_ID 
1 IT Billing 1 
2 Engineerin 2 
3 Finance 7 


交叉 连接 - CROSS JOIN 
交叉 连接 (CROSS JOIN) 把 第 一 个 表 的 每 一 行 与 第 二 个 表 的 每 一 行进 行 匹配 。 如 


果 两 个 输入 表 分 别 有 x 和 y 列 ， 则 结果 表 有 x+y 列 。 由 于 交叉 连接 (CROSS 
JOIN) 有 可 能 产生 非常 大 的 表 ， 使 用 时 必须 着 愤 ， 只 在 适当 的 时 候 使 用 它们 。 


下 面 是 交叉 连接 (CROSS JOIN) 的 语法 : 


SELECT ... FROM table1 CROSS JOIN table2 ... 


基于 上 面 的 表 ， 我 们 可 以 写 一 个 交叉 连接 (CROSS JOIN) ， 如 下 所 示 : 


sqlite> SELECT EMP_ID, NAME, DEPT FROM COMPANY CROSS JOIN DEPARTMEI 
Aoo ë 
上 面 的 查询 会 产生 以 下 结果 : 





1 Paul IT Billing 
2 Paul Engineerin 
7 Paul Finance 
1 Allen IT Billing 
2 Allen Engineerin 
7 Allen Finance 
1 Teddy IT Billing 
2 Teddy Engineerin 
7 Teddy Finance 
1 Mark IT Billing 
2 Mark Engineerin 
7 Mark Finance 
1 David IT Billing 
2 David Engineerin 
7 David Finance 
1 Kim IT Billing 
2 Kim Engineerin 
7 Kim Finance 
1 James IT Billing 
2 James Engineerin 
7 James Finance 


内 连接 -INNER JOIN 


内 连接 (INNER JOIN) 根据 连接 谓词 结合 两 个 表 (table1 和 table2) 的 列 值 来 创 
建 一 个 新 的 结果 表 。 查 询 会 把 table1 中 的 每 一 行 与 table2 中 的 每 一 行进 行 比较 ， 
找到 所 有 满足 连接 谓词 的 行 的 匹配 对 。 当 满足 连接 谓词 时 ，A 和 B 行 的 每 个 匹配 对 
的 列 值 会 合并 成 一 个 结果 行 。 


内 连接 (INNER JOIN)〉 是 最 常见 的 连接 类 型 ， 是 默认 的 连接 类 型 。INNER 关键 字 
是 可 选 的 。 


下 面 是 内 连接 (INNER JOIN) 的 语法 : 


SELECT ... FROM table1 [INNER] JOIN table2 ON conditional_expressic 





为 了 避免 元 余 ， 并 保持 较 短 的 措辞 ， 可 以 使 用 USING 表达 式 声 明 内 连接 (INNER 
JOIN) 条 件 。 这 个 表达 式 指 定 一 个 或 多 个 列 的 列表 : 


SELECT ... FROM table1 JOIN table2 USING ( columni ,... ) ... 


自然 连接 (NATURAL JOIN) 类 似 于 JOIN...USING， 只 是 它 会 自动 测试 存在 两 个 
表 中 的 每 一 列 的 值 之 间 相 等 值 : 


SELECT ... FROM table1 NATURAL JOIN table2... 


基于 上 面 的 表 ， 我 们 可 以 写 一 个 内 连接 (INNER JOIN) ， 如 下 所 示 : 


sqlite> SELECT EMP_ID, NAME, DEPT FROM COMPANY INNER JOIN DEPARTME? 
ON COMPANY.ID = DEPARTMENT ,EMP_ID ， 








‘| — 
上 面 的 查询 会 产生 以 下 结果 : 

EMP_ID NAME DEPT 

1 Paul IT Billing 

2 Allen Engineerin 

7 James Finance 


外 连接 - OUTER JOIN 


外 连接 (OUTER JOIN) 是 内 连接 (INNER JOIN) 的 扩展 。 虽 然 SQL 标准 定义 了 
三 种 类 型 的 外 连接 : LEFT、RIGHT、FULL， 但 SQLite 只 支持 左 外 连接 (LEFT 
OUTER JOIN) 。 


外 连接 (OUTER JOIN) 声明 条 件 的 方法 与 内 连接 (INNER JOIN) 是 相同 的 ， 使 
FA ON, USING 或 NATURAL 关键 字 来 表达 。 最 初 的 结果 表 以 相同 的 方式 进行 计 

算 。 一 旦 主 连接 计算 完成 ， 外 连接 (OUTER JOIN) 闻 从 一 个 或 两 个 表 中 任何 未 连 
接 的 行 合 并 进来 ， 外 连接 的 列 使 用 NULL 值 ， 将 它们 附加 到 结果 表 中 。 


下 面 是 左 外 连接 (LEFT OUTER JOIN) 的 语法 : 
SELECT ... FROM table1 LEFT OUTER JOIN table2 ON conditional_expre: 
I) 


为 了 避免 元 余 ， 并 保持 较 短 的 措辞 ， 可 以 使 用 USING 表达 式 声 明 外 连接 (OUTER 
JOIN) 条 件 。 这 个 表达 式 指 定 一 个 或 多 个 列 的 列表 : 





SELECT ... FROM table1 LEFT OUTER JOIN table2 USING ( column1 ,... 
基于 上 面 的 表 ， 我 们 可 以 写 一 个 外 连接 (OUTER JOIN) ， 如 下 所 示 : 





sqlite> SELECT EMP_ID, NAME, DEPT FROM COMPANY LEFT OUTER JOIN DEP/ 
ON COMPANY.ID = DEPARTMENT ,EMP_ID ， 


«| = 











上 面 的 查询 会 产生 以 下 结果 : 
EMP_ID NAME DEPT 
1 Paul IT Billing 
2 Allen Engineerin 
Teddy 
Mark 
David 
Kim 


7 James Finance 


SQLite Unions 子 句 


SQLite 的 UNION 子 句 /运算 符 用 于 合并 两 个 或 多 个 SELECT 语句 的 结果 ， 不 返回 
任何 重复 的 行 。 


为 了 使 用 UNION， 每 个 SELECT 被 选择 的 列 数 必须 是 相同 的 ， 相 同 数目 的 列表 达 
式 ， 相 同 的 数据 类 型 ， 并 确保 它们 有 相同 的 顺序 ， 但 它们 不 必 具 有 相同 的 长 度 。 


语法 
UNION 的 基本 语法 如 下 : 


SELECT columni [, column2 ] 
FROM table1 [, table2 ] 
[WHERE condition] 

UNION 

SELECT columni [, column2 ] 


FROM table1 [, table2 ] 
[WHERE condition] 


这 里 给 定 的 条 件 根据 需要 可 以 是 任何 表达 式 。 


实例 
假设 有 下 面 两 个 表 ， (1) COMPANY 表 如 下 所 示 : 


sqlite> select * from COMPANY; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000 .0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 





(2) 另 一 个 表 是 DEPARTMENT， 如 下 所 示 : 


1 IT Billing all 
2 Engineering 2 
3 Finance 7 
4 Engineering 3 
5 Finance 4 
6 Engineering 5 
7 Finance 6 


现在 ， 让 我 们 使 用 SELECT 语句 及 UNION 子 句 来 连接 两 个 表 ， 如 下 所 示 : 


sqlite> SELECT EMP_ID, NAME, DEPT FROM COMPANY INNER JOIN DEPARTME? 
ON COMPANY.ID = DEPARTMENT .EMP_ID 
UNION 
SELECT EMP_ID, NAME, DEPT FROM COMPANY LEFT OUTER JOIN DEPARTI 
ON COMPANY.ID = DEPARTMENT.EMP_ID,; 


TT 
这 将 产生 以 下 结果 : 





EMP_ID NAME DEPT 

1 Paul IT Billing 
2 Allen Engineerin 
3 Teddy Engineerin 
4 Mark Finance 

5 David Engineerin 
6 Kim Finance 

7 James Finance 

UNION ALL 子 句 


UNION ALL 运算 符 用 于 结合 两 个 SELECT 语句 的 结果 ， 包 括 重 复 行 。 
适用 于 UNION 的 规则 同样 适用 于 UNION ALL 运算 符 。 


语法 


UNION ALL 的 基本 语法 如 下 : 


SELECT columni [, column2 ] 
FROM table1 [, table2 ] 
[WHERE condition] 


UNION ALL 
SELECT columni [, column2 ] 


FROM tablei [, table2 ] 
[WHERE condition] 


这 里 给 定 的 条 件 根据 需要 可 以 是 任何 表达 式 。 


实例 


现在 ， 让 我 们 使 用 SELECT 语句 及 UNION ALL 子 句 来 连接 两 个 表 ， 如 下 所 示 : 


sqlite> SELECT EMP_ID, NAME, DEPT FROM COMPANY INNER JOIN DEPARTME? 


ON COMPANY.ID = DEPARTMENT .EMP_ID 


UNION ALL 


SELECT EMP_ID, NAME, DEPT FROM COMPANY LEFT OUTER JOIN DEPARTI 


ON COMPANY.ID = DEPARTMENT ,EMP_ID ， 


BE 


这 将 产生 以 下 结果 : 


NO OBWNENOOBRWNE 
a 
feb) 
3 
CD 
n 


IT Billing 
Engineerin 
Engineerin 
Finance 
Engineerin 
Finance 
Finance 
IT Billing 
Engineerin 
Engineerin 
Finance 
Engineerin 
Finance 
Finance 





SQLite NULL 值 


SQLite 的 NULL 是 用 来 表示 一 个 缺失 值 的 项 。 表 中 的 一 个 NULL 值 是 在 字段 中 显 
示 为 空白 的 一 个 值 。 


带 有 NULL 值 的 字段 是 一 个 不 带 有 值 的 字段 。NULL 值 与 需 值 或 包含 空格 的 字段 是 
不 同 的 ， 理 解 这 点 是 非常 重要 的 。 


语法 
创建 表 时 使 用 NULL 的 基本 语法 如 下 : 


SQLite> CREATE TABLE COMPANY (人 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL 


); 
在 这 里 ，NOT NULL 表示 列 总 是 接受 给 定数 据 类 型 的 显 式 值 。 这 里 有 两 个 列 我 们 没 
有 使 用 NOT NULL， 这 意味 着 这 两 个 列 不 能 为 NULL. 
带 有 NULL 值 的 字段 在 记录 创建 的 时 候 可 以 保留 为 空 。 


实例 


NULL 值 在 选择 数据 时 会 引起 问题 ， 因 为 当 把 一 个 未 知 的 值 与 另 一 个 值 进行 比较 
时 ， 结 果 总 是 未 知 的 ， 且 不 会 包含 在 最 后 的 结果 中 。 假 设 有 下 面 的 表 , COMPANY 
的 记录 如 下 所 示 : 


ID NAME AGE ADDRESS SALARY 

1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


让 我 们 使 用 UPDATE 语句 来 设置 一 些 允 许 空 值 的 值 为 NULL， 如 下 所 示 : 


sqlite> UPDATE COMPANY SET ADDRESS = NULL, SALARY = NULL where ID : 
‘| = a: 
现在 ，COMPANY 表 的 记录 如 下 所 示 : 








ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 

7 James 24 


接 下 来 ， 让 我 们 看 看 IS NOT NULL 运算 符 的 用 法 ， 它 用 来 列 出 所 有 SALARY 不 为 
NULL 的 记录 : 


sqlite> SELECT ID, NAME, AGE, ADDRESS, SALARY 
FROM COMPANY 
WHERE SALARY IS NOT NULL; 


上 面 的 SQLite 语句 将 产生 下 面 的 结果 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


下 面 是 IS NULL 运算 符 的 用 法 ， 将 列 出 所 有 SALARY 为 NULL 的 记录 : 


sqlite> SELECT ID, NAME, AGE, ADDRESS, SALARY 
FROM COMPANY 
WHERE SALARY IS NULL; 


上 面 的 SQLite 语句 将 产生 下 面 的 结果 : 


ADDRESS 


SALARY 


SQLite 别名 
您 可 以 暂时 把 表 或 列 重 命名 为 另 一 个 名 字 ， 这 被 称 为 别名 。 使 用 表 别 名 是 指 在 一 个 


特定 的 SQLite 语句 中 重 命名 表 。 重 命名 是 临时 的 改变 ， 在 数据 库 中 实际 的 表 的 名 
称 不 会 改变 。 


列 别 名 用 来 为 某 个 特定 的 SQLite 语句 重 命 名 表 中 的 列 。 
语法 
R 别名 的 基本 语法 如 下 : 


SELECT columni, column2.... 
FROM table_name AS alias_name 
WHERE [condition]; 


列 别名 的 基本 语法 如 下 : 


SELECT column_name AS alias_name 
FROM table_name 
WHERE [condition]; 


实例 
假设 有 下 面 两 个 表 ， (1) COMPANY 表 如 下 所 示 : 


sqlite> select * from COMPANY; 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 





(2) 另 一 个 表 是 DEPARTMENT， 如 下 所 示 : 


1 IT Billing all 
2 Engineering 2 
3 Finance 7 
4 Engineering 3 
5 Finance 4 
6 Engineering 5 
7 Finance 6 


mE, FHE 表 别 名 的 用 法 ， 在 这 里 我 们 使 用 C AD 分 别 作为 COMPANY 和 
DEPARTMENT 表 的 别名 : 


sqlite> SELECT C.ID, C.NAME, C.AGE, D.DEPT 
FROM COMPANY AS C, DEPARTMENT AS D 
WHERE C.ID = D.EMP_ID; 


上 面 的 SQLite 语句 将 产生 下 面 的 结果 : 


ID NAME AGE DEPT 

1 Paul 32 IT Billing 
2 Allen 25 Engineerin 
3 Teddy 23 Engineerin 
4 Mark 25 Finance 

5 David 27 Engineerin 
6 Kim 22 Finance 

7 James 24 Finance 


让 我 们 看 一 个 列 别名 的 实例 ， 在 这 里 COMPANY _ID 是 ID 列 的 别名 ， 
COMPANY_NAME Æ name 列 的 别名 : 


sqlite> SELECT C.ID AS COMPANY_ID, C.NAME AS COMPANY_NAME, C.AGE, I 
FROM COMPANY AS C, DEPARTMENT AS D 
WHERE C.ID = D.EMP_ID,; 


EE 
上 面 的 SQLite 语句 将 产生 下 面 的 结果 : 





COMPANY_ID COMPANY_NAME AGE DEPT 


1 Paul 32 IT Billing 
2 Allen 25 Engineerin 
3 Teddy 23 Engineerin 
4 Mark 25 Finance 
5 David 27 Engineerin 
6 Kim 22 Finance 
7 James 24 Finance 


SQLite 触发 右 (Trigger) 


SQLite 的 触发 器 是 数据 库 的 回调 函数 ， 它 会 自动 执行 /指定 的 数据 库 事件 发 生 时 调 


用 。 


AÉ 


以 下 是 关于 SQLite 的 触发 器 的 要 点 : SQLite 触发 器 (Trigger) 是 数据 库 的 回 
数 ， 它 会 在 指定 的 数据 库 事件 发 生 时 自动 执行 /调用 。 以 下 是 关于 SQLite BAR 


发 器 (Trigger) 的 要 点 : 


Bf 


Ta / 


SQLite 的 触发 器 (Trigger) 可 以 指定 在 特定 的 数据 库 表 发 生 DELETE, 
INSERT 或 UPDATE 时 触发 ， 或 在 一 个 或 多 个 指定 表 的 列 发 生 更 新 时 触发 。 


SQLite 只 支持 FOR EACH ROW 触发 器 (Trigger) ， 没 有 FOR EACH 
STATEMENT 触发 器 (Trigger) 。 因 此 ， 明 确 指定 FOR EACH ROW 是 可 选 
的 。 


WHEN 子 句 和 人 触发 器 (Trigger) 动作 可 能 访问 使 用 表单 NEW.column-name 
和 OLD.column-name 的 引用 插入 、 删 除 或 更 新 的 行 元 素 ， 其 中 column- 
name 是 从 与 触发 器 关联 的 表 的 列 的 名 称 。 


如 果 提 供 WHEN 子 句 ， 则 只 针对 WHEN 子 句 为 真 的 指定 行 执行 SQL 语句 。 
如 果 没 有 提供 WHEN 子 句 ， 则 针对 所 有 行 执 行 SQL 语句 。 


BEFORE 或 AFTER 关键 字 决 定 何 时 执行 触发 器 动作 ， 决 定 是 在 关联 行 的 插 
入 、 修 改 或 删除 之 前 或 者 之 后 执行 触发 器 动作 。 


当 触 发 器 相关 联 的 表 删 除 时 ， 自 动 删 除 触 发 器 (Trigger) o 


要 修改 的 表 必 须 存 在 于 同一 数据 库 中 ， 作 为 触发 器 被 附加 的 表 或 视图 ， 且 必须 
只 使 用 tablename， 而 不 是 database.tablename。 


一 个 特殊 的 SQL 函数 RAISE() 可 用 于 触发 器 程序 内 抛 出 异常 。 


创建 触发 器 (Trigger) 的 基本 语法 如 下 : 


CREATE TRIGGER trigger_name [BEFORE|AFTER] event_name 
ON table_name 
BEGIN 


- Trigger logic goes here.... 


END; 


在 这 里 ，event_name 可 以 是 在 所 提 到 的 表 table_name 上 的 INSERT, DELETE 
和 UPDATE 数据 库 操作 。 您 可 以 在 表 名 后 选择 指定 FOR EACH ROW. 


4 


以 下 是 在 UPDATE 操作 上 在 表 的 一 个 或 多 个 指定 列 上 创建 触发 器 (Trigger) 的 语 


CREATE TRIGGER trigger_name [BEFORE|AFTER] UPDATE OF column_name 
ON table_name 
BEGIN 
-- Trigger logic goes here.... 
END; 


4 | 
实例 


让 我 们 假设 一 个 情况 ， 我 们 要 为 被 插入 到 新 创建 的 COMPANY R (如 果 已 经 存 
在 ， 则 删除 重新 创建 ) 中 的 每 一 个 记录 保持 审计 试验 : 





sqlite> CREATE TABLE COMPANY( 


ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL 


); 


为 了 保持 审计 试验 ， 我 们 将 创建 一 个 名 为 AUDIT 的 新 表 。 每 当 COMPANY 表 中 有 
一 个 新 的 记录 项 时 ， 日 志 消 息 将 被 插入 其 中 : 


sqlite> CREATE TABLE AUDIT( 
EMP_ID INT NOT NULL, 
ENTRY_DATE TEXT NOT NULL 


); 


在 这 里 ，ID Æ AUDIT 记录 的 ID，EMP ID 2328 COMPANY 349 ID, DATE 将 
保持 COMPANY 中 记录 被 创建 时 的 时 间 戳 。 所 以 ， 现 在 让 我 们 在 COMPANY RE 
创建 一 个 触发 器 ， 如 下 所 示 : 


sqlite> CREATE TRIGGER audit_log AFTER INSERT 
ON COMPANY 
BEGIN 
INSERT INTO AUDIT(EMP_ID, ENTRY_DATE) VALUES (new.ID, datetime( 
END; 


图 一 


现在 ， 我 们 将 开始 在 COMPANY 表 中 插入 记录 ， 这 将 导致 在 AUDIT 表 中 创建 一 个 
审计 日 志 记 录 。 因 此 ， 让 我 们 在 COMPANY 表 中 创建 一 个 记录 ， 如 下 所 示 : 





sqlite> INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (1, 'Paul', 32, 'California', 20000.00 ); 


这 将 在 COMPANY 表 中 创建 如 下 一 个 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 


同时 ， 将 在 AUDIT 表 中 创建 一 个 记录 。 这 个 纪录 是 触发 器 的 结果 ， 这 是 我 们 在 
COMPANY REJ INSERT 操作 上 创建 的 触发 器 (Trigger) 。 类 似 的 ， 可 以 根据 
需要 在 UPDATE 和 DELETE 操作 上 创建 触发 器 (Trigger) 。 


EMP_ID ENTRY_DATE 


1 2013-04-05 06:26:00 


列 出 触发 器 (TRIGGERS) 
您 可 以 从 sqlite_master 表 中 列 出 所 有 触发 器 ， 如 下 所 示 : 


sqlite> SELECT name FROM sqlite master 
WHERE type = 'trigger'; 


上 面 的 SQLite 语句 只 会 列 出 一 个 条 目 ， 如 下 : 


audit_log 


如 果 您 想 要 列 出 特定 表 上 的 触发 器 ， 则 使 用 AND 子 句 连接 表 名 ， 如 下 所 示 : 


sqlite> SELECT name FROM sqlite master 
WHERE type = 'trigger' AND tbl name = 'COMPANY'; 


上 面 的 SQLite 语句 只 会 列 出 一 个 条 目 ， 如 下 : 


audit_log 


删除 触发 器 (TRIGGERS) 
下 面 是 DROP 命令， 可 用 于 删除 已 有 的 触发 器 : 


sqlite> DROP TRIGGER trigger_name; 


SQLite 5| (Index) 


索引 (Index) 是 一 种 特殊 的 查找 表 ， 数 据 库 搜索 引擎 用 来 加 快 数据 检索 。 简 单 地 
说 eo ese eee 一 个 数据 库 中 的 索引 与 一 本 书后 边 的 索引 是 
非常 相似 的 。 


例如 ， 如 果 您 想 在 一 本 讨论 某 个 话题 的 书 中 引用 所 有 页 面 ， 您 首先 需要 指向 索引 ， 
索引 按 字母 顺序 列 出 了 所 有 主题 ， 然 后 指向 一 个 或 多 个 特定 的 页 码 。 


索引 有 助 于 加 快 SELECT 查询 和 WHERE 子 句 ， 但 它 会 减 慢 使 用 UPDATE 和 
INSERT 语句 时 的 数据 和 输入。 索引 可 以 创建 或 删除 ， 但 不 会 影响 数据 。 


使 用 CREATE INDEX 语句 创建 索引 ， 它 允许 命名 索引 ， 指 定 表 及 要 索引 的 一 列 或 
多 列 ， 并 指示 索引 是 升序 排列 还 是 降序 排列 。 


索引 也 可 以 是 唯一 的 ， 和 与 UNIQUE 约束 类 似 ， 在 列 上 或 列 组 合 上 防止 重复 条 目 。 


CREATE INDEX AS 
CREATE INDEX 的 基本 语法 如 下 : 


CREATE INDEX index_name ON table_name; 


单列 索引 
单列 索引 是 一 个 只 基于 表 的 一 个 列 上 创建 的 索引 。 基 本 语法 如 下 : 


CREATE INDEX index_name 
ON table_name (column_name); 


ME— 325 | 


使 用 唯一 索引 不 仅 是 为 了 性 能 ， 同 时 也 为 了 数据 的 完整 性 。 唯 一 索引 不 允许 任何 重 
复 的 值 插入 到 表 中 。 基 本 语法 如 下 : 


CREATE INDEX index_name 
on table_name (column_name); 


BERKI 


组 合 索引 是 基于 一 个 表 的 两 个 或 多 个 列 上 创建 的 索引。 基本 语法 如 下 : 


CREATE INDEX index_name 
on table_name (columni, column2); 


是 否 要 创建 一 个 单列 素 引 还 是 组 合 素 引 ， 要 考虑 到 您 在 作为 查询 过 小 条 件 的 
WHERE 子 句 中 使 用 非常 频繁 的 列 。 


如 果 值 使 用 到 一 个 列 ， 则 选择 使 用 单列 索引 。 如 果 在 作为 过 滤 的 WHERE 子 句 中 有 
两 个 或 多 个 列 经 常 使 用 ， 则 选择 使 用 组 合 索引 。 


隐 式 索引 


隐 式 索引 是 在 创建 对 象 时 ， 由 数据 库 服务 器 自动 创建 的 索引 。 索 引 自动 创建 为 主键 
约束 和 唯一 约束 。 


实例 
下 面 是 一 个 例子 ， 我 们 将 在 COMPANY RHY salary 列 上 创建 一 个 索引 : 


sqlite> CREATE INDEX salary_index ON COMPANY (salary); 


现在 ， 让 我 们 使 用 .indices 命令 列 出 COMPANY 表 上 所 有 可 用 的 索引 ， 如 下 所 
ZN : 


sqlite> .indices COMPANY 


这 将 产生 如 下 结果 ， 其 中 sqlite_autoindex. COMPANY 1 是 创建 表 时 创建 的 隐 式 
RA | o 


salary_index 
sqlite_autoindex_COMPANY_1 


您 可 以 列 出 数据 库 范 围 的 所 有 索引 ， 如 下 所 示 : 


sqlite> SELECT * FROM sqlite master WHERE type = 'index'; 


DROP INDEX 命令 


一 个 索引 可 以 使 用 SQLite 的 DROP 命令 删除 。 当 删除 索引 时 应 特别 注意 ， 因 为 性 


能 可 能 会 下 降 或 提高 。 


基本 语法 如 下 : 


DROP INDEX index_name; 


您 可 以 使 用 下 面 的 语句 来 删除 之 前 创建 的 索引 : 


sqlite> DROP INDEX salary_index; 


什么 情况 下 要 避免 使 用 索引 ? 


虽然 索引 的 目的 在 于 提高 数据 库 的 性 能 ， 但 这 里 有 几 个 情况 需要 避免 使 用 索引 。 使 
用 索引 时 ， 应 重新 考虑 下 列 准则 : 


。 索引 不 应 该 使 用 在 较 小 的 表 上 。 

。 索引 不 应 该 使 用 在 有 频繁 的 大 批量 的 更 新 或 插入 操作 的 表 上 。 
。 索引 不 应 该 使 用 在 含有 大 量 的 NULL 值 的 列 上 。 

。 索引 不 应 该 使 用 在 频繁 操作 的 列 上 。 


SQLite Indexed By 


"INDEXED BY index-name" 子 句 规定 必须 需要 命名 的 索引 来 查找 前 面 表 中 值 。 
如 果 索 引 名 index-name 不 存在 或 不 能 用 于 查询 ， 然 后 SQLite 语句 的 准备 失败 。 


"NOT INDEXED" 子 句 规定 当 访 问 前 面 的 表 (包括 由 UNIQUE 和 PRIMARY KEY 
约束 创建 的 隐 式 索引 ) 时 ， 没 有 使 用 索引 。 


然而 ， 即 使 指定 了 "NOT INDEXED", INTEGER PRIMARY KEY 仍然 可 以 被 用 于 
查找 条 目 。 


语法 


下 面 是 INDEXED BY 子 句 的 语法 ， 它 可 以 与 DELETE、UPDATE 或 SELECT 语 
句 一 起 使 用 : 


SELECT | DELETE | UPDATE columni, column2... 
INDEXED BY (index_name) 

table_name 

WHERE (CONDITION); 


实例 
假设 有 表 COMPANY， 我 们 将 创建 一 个 索引 ， 并 用 它 进行 INDEXED BY 操作 。 


sqlite> CREATE INDEX salary_index ON COMPANY(salary); 
sqlite> 


现在 使 用 INDEXED BY +4) M3 COMPANY 中 选择 数据 ， 如 下 所 示 : 


sqlite> SELECT * FROM COMPANY INDEXED BY salary_index WHERE salary 
H — = 





SQLite Alter MS 


SQLite 的 ALTER TABLE 命令 不 通过 执行 一 个 完整 的 转 储 和 数据 的 重 载 来 修改 已 
有 的 表 。 您 可 以 使 用 ALTER TABLE 语句 重 命名 表 ， 使 用 ALTER TABLE 语句 还 可 
以 在 已 有 的 表 中 添加 额外 的 列 。 


在 SQLite 中 ， 除 了 重 命名 表 和 在 已 有 的 表 中 添加 列 ，ALTER TABLE 命令 不 支持 
其 他 操作 。 


语法 
用 来 重 命名 已 有 的 表 的 ALTER TABLE 的 基本 语法 如 下 : 


ALTER TABLE database_name.table_name RENAME TO new_table_name; 


用 来 在 已 有 的 表 中 添加 一 个 新 的 列 的 ALTER TABLE 的 基本 语法 如 下 : 


ALTER TABLE database_name.table_name ADD COLUMN column_def...; 


实例 


假设 我 们 的 COMPANY 表 有 如 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000 .0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


现在 ， 让 我 们 党 试 使 用 ALTER TABLE 洛 句 重 命名 该 表 ， 如 下 所 示 : 


sqlite> ALTER TABLE COMPANY RENAME TO OLD_COMPANY; 


上 面 的 SQLite 语句 将 重 命名 COMPANY RA OLD_COMPANY。 现 在 ， 让 我 们 党 
RE OLD_COMPANY 表 中 添加 一 个 新 的 列 ， 如 下 所 示 : 


sqlite> ALTER TABLE OLD_COMPANY ADD COLUMN SEX char(1); 


现在 ，COMPANY 表 已 经 改变 ， 使 用 SELECT 语句 输出 如 下 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


请 注意 ， 新 添加 的 列 是 以 NULL 值 来 填充 的 。 


SQLite Truncate Table 


在 SQLite 中 ， 并 没有 TRUNCATE TABLE AS, (ELA SQLite 的 DELETE 
命令 从 已 有 的 表 中 删除 全 部 的 数据 ， 但 建议 使 用 DROP TABLE AA BPRE TR, 
然后 再 重新 创建 一 通 。 


语法 
DELETE 命令 的 基本 语法 如 下 : 


sqlite> DELETE FROM table_name; 


DROP TABLE 的 基本 语法 如 下 : 


sqlite> DROP TABLE table_name; 


如 果 您 使 用 DELETE TABLE 命令 删除 所 有 记录 ， 建 议 使 用 VACUUM 命令 清除 未 
使 用 的 空间 。 


实例 


假设 COMPANY 表 有 如 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


下 面 为 删除 上 表 记 录 的 实例 : 


SQLite> DELETE FROM COMPANY; 
SQLite> VACUUM; 


mI, COMPANY 表 中 的 记录 完全 被 删除 ， 使 用 SELECT 话 句 将 没有 任何 输出 。 


SQLite 视图 (View) 
视图 (View) 只 不 过 是 通过 相关 的 名 称 存 储 在 数据 库 中 的 一 个 SQLite 语句 。 视 图 
(View) 实际 上 是 一 个 以 预定 义 的 SQLite 查询 形式 存在 的 表 的 组 合 。 


视图 View) 可 以 包含 一 个 表 的 所 有 行 或 从 一 个 或 多 个 表 选 定 行 。 视 图 (View) 可 
以 从 一 个 或 多 个 表 创 建 ， 这 取决 于 要 创建 视图 的 SQLite 查询 。、 


视图 (View) 是 一 种 虚 表 ， 人 允许 用 户 实现 以 下 几 点 : 

。 用 户 或 用 户 组 查找 结构 数据 的 方式 更 自然 或 直观 。 

。 限制 数据 访问 ， 用 户 只 能 看 到 有 限 的 数据 ， 而 不 是 完整 的 表 。 
。 汇总 各 种 表 中 的 数据 ， 用 于 生成 报告 。 


SQLite 视图 是 只 读 的 ， 因 此 可 能 无 法 在 视图 上 执行 DELETE, INSERT 或 
UPDATE 语句 。 但 是 可 以 在 视图 上 创建 一 个 触发 器 ， 当 党 斌 DELETE, INSERT 或 
UPDATE 视图 时 触发 ， 需 要 做 的 动作 在 触发 器 内 容 中 定义 。 


创建 视图 


SQLite 的 视图 是 使 用 CREATE VIEW 语句 创建 的 。SQLite 视图 可 以 从 一 个 单一 的 
表 、 多 个 表 或 其 他 视图 创建 。 


CREATE VIEW 的 基本 语法 如 下 : 
CREATE [TEMP | TEMPORARY] VIEW view_name AS 
SELECT columni, column2..... 


FROM table_name 
WHERE [condition]; 


您 可 以 在 SELECT 语句 中 包含 多 个 表 ， 这 和 与 在 正常 的 SQL SELECT 查询 中 的 方式 
非常 相似 。 如 果 使 用 了 可 选 的 TEMP 或 TEMPORARY 关键 字 ， 则 将 在 临时 数据 库 
中 创建 视图 。 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


现在 ， 下 面 是 一 个 从 COMPANY 表 创 建 视 图 的 实例 。 视 图 只 从 COMPANY 表 中 选 
取 几 列 : 


sqlite> CREATE VIEW COMPANY_VIEW AS 
SELECT ID, NAME, AGE 
FROM COMPANY; 


现在 ， 可 以 查询 COMPANY VIEW， 与 查询 实际 表 的 方式 类 似 。 下 面 是 实例 : 


sqlite> SELECT * FROM COMPANY_VIEw; 


这 将 产生 以 下 结果 
ID NAME AGE 
iL Paul 32 
2 Allen 25 
3 Teddy 23 
4 Mark 25 
5 David 27 
6 Kim 22 
7 James 24 

删除 视图 


要 删除 视图 ， 只 需 使 用 带 有 view_name 的 DROP VIEW 语句 。DROP VIEW 的 基 
本 语法 如 下 : 


sqlite> DROP VIEW view_name; 


下 面 的 命令 将 删除 我 们 在 前 面 创建 的 COMPANY_VIEW 视图 : 


sqlite> DROP VIEW COMPANY_VIEw; 


SQLite 事务 (Transaction) 


事务 (Transaction) 是 一 个 对 数据 库 执 行 工 作 单元 。 事 务 (Transaction) 是 以 逻辑 
顺序 完成 的 工作 单位 或 序列 ， 可 以 是 由 用 户 手 动 操作 完成 ， 也 可 以 是 由 某 种 数据 库 
程序 自动 完成 。 


事务 (Transaction) 是 指 一 个 或 多 个 更 改 数据 库 的 扩展 。 例 如 ， 如 果 您 正在 创建 一 
个 记录 或 者 更 新 一 个 记录 或 者 从 表 中 删除 一 个 记录 ， 那 么 您 正在 该 表 上 执行 事务 。 
重要 的 是 要 控制 事务 以 确保 数据 的 完整 性 和 处理 数 据 库 错误 。 


实际 上 ， 您 可 以 把 许多 的 SQLite 查询 联合 成 一 组 ， 把 所 有 这 些 放 在 一 起 作为 事务 
的 一 部 分 进行 执行 。 


事务 的 属性 
事务 (Transaction) 具有 以 下 四 个 标准 属性 ， 通 常 根据 首 字 母 缩写 为 ACID : 


。 原子 性 (Atomicity) : 确保 工作 单位 内 的 所 有 操作 都 成 功 完 成 ， 否 则 ， 事 务 
会 在 出 现 故 障 时 终止 ， 之 前 的 操作 也 会 回 滚 到 以 前 的 状态 。 


e —3 (Consistency) : 确保 数据 库 在 成 功 提交 的 事务 上 正确 地 改变 状态 。 
e 隔离 性 (Isolation) : 使 事务 操作 相互 独立 和 透明 。 


e AlE (Durability) : 确保 已 提交 事务 的 结果 或 效果 在 系统 发 生 故 障 的 情况 
下 仍然 存在 。 


事务 控制 


使 用 下 面 的 命令 来 控制 事务 : 
e BEGIN TRANSACTION : 开始 事务 处 理 。 
e COMMIT: 保存 更 改 ， 或 者 可 以 使 用 END TRANSACTION AP. 
。 ROLLBACK : 回 滚 所 做 的 更 改 。 


事务 控制 命令 只 与 DML AS INSERT, UPDATE 和 DELETE 一 起 使 用 。 他 们 不 能 
在 创建 表 或 删除 表 时 使 用 ， 因 为 这 些 操作 在 数据 库 中 是 自动 提交 的 。 


BEGIN TRANSACTION 命令 


事务 (Transaction) 可 以 使 用 BEGIN TRANSACTION 命令 或 简单 的 BEGIN AS 
来 启动 。 此 类 事务 通常 会 持续 执行 下 去 ， 直 到 遇 到 下 一 个 COMMIT 或 ROLLBACK 
命令 。 不 过 在 数据 库 关 闭 或 发 生 错误 时 ， 事 务 处 理 也 会 回 滚 。 以 下 是 启动 一 个 事务 


的 简单 语法 : 
BEGIN; 


or 


BEGIN TRANSACTION, 


COMMIT 命 兮 


COMMIT 命令 是 用 于 把 事务 调用 的 更 改 保存 到 数据 库 中 的 事务 命令 。 


COMMIT 命令 把 自 上 次 COMMIT 或 ROLLBACK 命令 以 来 的 所 有 事务 保存 到 数据 
库 。 


COMMIT 命令 的 语法 如 下 : 
COMMIT; 
or 


END TRANSACTION, 


ROLLBACK AS 

ROLLBACK 命令 是 用 于 撤消 尚未 保存 到 数据 库 的 事务 的 事务 命令 。 

ROLLBACK 命令 只 能 用 于 撤销 自 上 次 发 出 COMMIT 或 ROLLBACK 命令 以 来 的 事 
务 。 


ROLLBACK 命令 的 语法 如 下 : 


ROLLBACK; 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


现在 ， 让 我 们 开始 一 个 事务 ， 并 从 表 中 删除 age = 25 的 记录 ， 最 后 ， 我 们 使 用 
ROLLBACK 命 使 撤消 所 有 的 更 改 。 


sqlite> BEGIN; 
sqlite> DELETE FROM COMPANY WHERE AGE = 25; 
sqlite> ROLLBACK; 


检查 COMPANY 表 ， 仍 然 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000 .0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


现在 ， 让 我 们 开始 另 一 个 事务 ， 从 表 中 删除 age = 25 的 记录 ， 最 后 我 们 使 用 
COMMIT 命令 提交 所 有 的 更 改 。 


sqlite> BEGIN; 
sqlite> DELETE FROM COMPANY WHERE AGE = 25; 
sqlite> COMMIT; 


检查 COMPANY 表 ， 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
3 Teddy 23 Norway 20000.0 
5 David 27 Texas 85000 .0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 
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SQLite 4 i4 

子 查 询 或 内 部 查询 或 藤 套 查询 是 在 另 一 个 SQLite AiQABRAZ WHERE 子 句 中 的 
查询 。 

使 用 子 查询 返回 的 数据 将 被 用 在 主 查 询 中 作为 条 件 ， 以 进一步 限制 要 检索 的 数据 。 


子 查询 可 以 与 SELECT、INSERT、UPDATE 和 DELETE 语句 一 起 使 用 ， 可 伴随 
着 使 用 运算 符 如 =、<、>、>=、<=、IN、BETWEEN 等 。 


以 下 是 子 查询 必须 遵循 的 几 个 规则 : 


子 查询 必须 用 括号 括 起 来 。 


子 查询 在 SELECT 子 句 中 只 能 有 一 个 列 ， 除 非 在 主 查 询 中 有 多 列 ， 和 与 子 查询 
的 所 选 列 进行 比较 。 


ORDER BY 不 能 用 在 子 查询 中 ， 虽 然 主 查询 可 以 使 用 ORDER BY。 可 以 在 子 
查询 中 使 用 GROUP BY， 功 能 与 ORDER BY 相同 。 


子 查询 返回 多 于 一 行 ， 只 能 与 多 值 运算 符 一 起 使 用 ， 如 IN 运算 符 。 


BETWEEN 运算 符 不 能 与 子 查 询 一 起 使 用 ， 但 是 ，BETWEEN 可 在 子 查 询 内 
使 用 。 


SELECT 语句 中 的 子 查 询 使 用 
子 查询 通常 与 SELECT 语句 一 起 使 用 。 基 本 语法 如 下 : 


SELECT column_name [, column_name ] 
FROM tablei [, table2 | 
WHERE column_name OPERATOR 


(SELECT column_name [, column_name ] 
FROM tablei [, table2 ] 
[WHERE] ) 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


现在 ， 让 我 们 检查 SELECT 语句 中 的 子 查询 使 用 : 


sqlite> SELECT * 
FROM COMPANY 
WHERE ID IN (SELECT ID 
FROM COMPANY 
WHERE SALARY > 45000) ; 


这 将 产生 以 下 结果 
ID NAME AGE ADDRESS SALARY 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 


INSERT 语句 中 的 子 查 询 使 用 


子 查询 也 可 以 与 INSERT 语句 一 起 使 用 。INSERT 语句 使 用 子 查询 返回 的 数据 插入 
到 另 一 个 表 中 。 在 子 查询 中 所 选择 的 数据 可 以 用 任何 字符 、 日 期 或 数字 函数 修改 。 


基本 语法 如 下 : 


INSERT INTO table name [ (column1i [, column2 ]) ] 
SELECT [ *|columni [, column2 ] 
FROM tablei [, table2 ] 
[ WHERE VALUE OPERATOR | 


例 


假设 COMPANY_BKP 的 结构 与 COMPANY 表 相 似 ， 且 可 使 用 相同 的 CREATE 
TABLE 进行 创建 ， 只 是 表 名 改 为 COMPANY_BKP。 现 在 把 整个 COMPANY RE 
制 到 COMPANY_BKP， 语 法 如 下 : 


将 


sqlite> INSERT INTO COMPANY_BKP 
SELECT * FROM COMPANY 
WHERE ID IN (SELECT ID 
FROM COMPANY) ; 


UPDATE 话 句 中 的 子 查询 使 用 


子 查询 可 以 与 UPDATE 语句 结合 使 用 。 当 通过 UPDATE 语句 使 用 子 查 询 时 ， 表 中 
单个 或 多 个 列 被 更 新 。 


基本 语法 如 下 : 


UPDATE table 
SET column_name = new_value 
[ WHERE OPERATOR [ VALUE ] 
(SELECT COLUMN_NAME 
FROM TABLE_NAME) 
[ WHERE) ] 


实例 


假设 ， 我 们 有 COMPANY_BKP 表 ， 是 COMPANY 表 的 各 份 。 


下 面 的 实例 把 COMPANY 表 中 所 有 AGE 大 于 或 等 于 27 的 客户 的 SALARY 更 新 为 
原来 的 0.50 倍 : 


sqlite> UPDATE COMPANY 
SET SALARY = SALARY * 0.50 
WHERE AGE IN (SELECT AGE FROM COMPANY_BKP 
WHERE AGE >= 27 ); 


这 将 影响 两 行 ， 最 后 COMPANY 表 中 的 记录 如 下 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 10000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 42500.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


DELETE 语句 中 的 子 查 询 使 用 


子 查询 可 以 与 DELETE 语句 结合 使 用 ， 就 像 上 面 提 到 的 其 他 语句 一 样 。 
基本 语法 如 下 : 
DELETE FROM TABLE_NAME 
[ WHERE OPERATOR [ VALUE ] 
(SELECT COLUMN_NAME 


FROM TABLE_NAME) 
[ WHERE) ] 


实例 


假设 ， 我 们 有 COMPANY_BKP 表 ， 是 COMPANY 表 的 各 份 。 
下 面 的 实例 删除 COMPANY 表 中 所 有 AGE 大 于 或 等 于 27 的 客户 记录 : 


sqlite> DELETE FROM COMPANY 
WHERE AGE IN (SELECT AGE FROM COMPANY_BKP 
WHERE AGE > 27 ); 


这 将 影响 两 行 ， 最 后 COMPANY 表 中 的 记录 如 下 : 


ID NAME AGE ADDRESS SALARY 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 42500.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


SQLite Autoincrement (自动 递增 ) 


SQLite 的 AUTOINCREMENT 是 一 个 关键 字 ， 用 于 表 中 的 字段 值 自动 递增 。 我 们 
可 以 在 创建 表 时 在 特定 的 列 名 称 上 使 用 AUTOINCREMENT 关键 字 实现 该 字段 值 的 
自动 增加 。 


关键 字 AUTOINCREMENT 只 能 用 于 整 型 (INTEGER) 字段 。 


语法 
AUTOINCREMENT 关键 字 的 基本 用 法 如 下 : 


CREATE TABLE table_name( 
columni INTEGER AUTOINCREMENT, 
column2 datatype, 
column3 datatype, 


columnN datatype, 
); 


实例 
假设 要 创建 的 COMPANY 表 如 下 所 示 : 


sqlite> CREATE TABLE COMPANY( 
ID INTEGER PRIMARY KEY AUTOINCREMENT, 


NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL 


); 


现在 ， 向 COMPANY 表 插 入 以 下 记录 : 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY) 
VALUES ( 'Paul', 32, 'California', 20000.00 ); 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY ) 
VALUES ('Allen', 25, 'Texas', 15000.00 ); 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY ) 
VALUES ('Teddy', 23, 'Norway', 20000.00 ); 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY) 
VALUES ( 'Mark', 25, 'Rich-Mond ', 65000.00 ); 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY) 
VALUES ( 'David', 27, 'Texas', 85000.00 ); 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY) 
VALUES ( 'Kim', 22, 'South-Hall', 45000.00 ); 


INSERT INTO COMPANY (NAME, AGE, ADDRESS, SALARY ) 
VALUES ( 'James', 24, 'Houston', 10000.00 ); 


这 将 向 COMPANY KHA 7 个 元 组 ， 此 时 COMPANY 表 的 记录 如 下 : 


ID NAME AGE ADDRESS SALARY 
1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


SQLite 注入 


如 果 您 的 站 点 允许 用 户 通过 网 页 输入 ， 并 将 输入 内 容 插入 到 SQLite 数据 库 中 ， 这 
个 时 候 您 就 面临 着 一 个 被 称 为 SQL 注入 的 安全 问题 。 本 章节 将 向 您 讲解 如 何 防止 
这 种 情况 的 发 生 ， 确 保 脚 本 和 SQLite 语句 的 安全 。 


注入 通常 在 请 求 用 户 输入 时 发 生 ， 比 如 需要 用 户 输入 姓名 ， 但 用 户 却 输入 了 一 个 
SQLite 语句 ， 而 这 语句 就 会 在 不 知 不 觉 中 在 数据 库 上 运行 。 


永远 不 要 相信 用 户 提供 的 数据 ， 所 以 只 处 理 通过 验证 的 数据 ， 这 项 规则 是 通过 模式 
匹配 来 完成 的 。 在 下 面 的 实例 中 ， 用 户 名 username 被 限制 为 字母 数字 字符 或 者 下 
IR, KELATE 8 到 20 个 字符 之 间 - 请 根据 需要 修改 这 些 规则 。 


if (preg_match("/^\w{8,20}$/", $_GET['username'], $matches)){ 

$db = new SQLiteDatabase('filename'); 

$result = @$db->query("SELECT * FROM users WHERE username=$matcl 
selse{ 

echo "username not accepted"; 
} 


二 = 





为 了 演示 这 个 问题 ， 假 设 考虑 此 摘录 : To demonstrate the problem, consider this 
excerpt: 


$name = "Qadir'; DELETE FROM users;"; 
@$db->query("SELECT * FROM users WHERE username='{$name}'") ; 


轴 数 调用 是 为 了 从 用 户 表 中 检索 name 列 与 用 户 指定 的 名 称 相 匹配 的 记录 。 正 常情 
况 下 ，$name 只 包含 字母 数字 字符 或 者 空格 ， 比 如 字符 串 ilia。 但 在 这 里 ， 向 
$name 追加 了 一 个 全 新 的 查询 ， 这 个 对 数据 库 的 调用 将 会 造成 灾难 性 的 问题 : 注入 
的 DELETE 查询 会 删除 users 的 所 有 记录 。 


虽然 已 经 存在 有 不 允许 查询 堆 生 或 在 单个 函数 调用 中 执行 多 个 查询 的 数据 库 接口 ， 
如 果 尝 试 堆 受 查询 ， 则 会 调用 失败 ， 但 SQLite 和 PostgreSQL 里 仍 进行 堆 受 查 
询 ， 即 执行 在 一 个 字符 串 中 提供 的 所 有 查询 ， 这 会 导致 严重 的 安全 问题 。 


防止 SQL 注入 


在 脚本 语言 中 ， 上 比如 PERL 和 PHP， 您 可 以 巧妙 地 处 理 所 有 的 转 义 字符 。 编 程 语 
言 PHP 提供 了 字符 串 函 数 sqlite_escape_string() 来 转 义 对 于 SQLite 来 说 比较 特 
殊 的 输入 字符 。 


if (get_magic_quotes_gpc()) 
$name = sqlite_escape_string($name) ; 


} 
$result = @$db->query("SELECT * FROM users WHERE username=' {$name} 
Soe > | 








q 





虽然 编码 使 得 插入 数据 变 得 安全 ， 但 是 它 会 呈现 简单 的 文本 比较 ， 在 查询 中 ， 对 于 
包含 二 进 制 数据 的 列 ，LIKE 子 句 是 不 可 用 的 。 

请 注意 ，addslashes() 不 应 该 被 用 在 SQLite 查询 中 引用 字符 串 ， 它 会 在 检索 数据 
时 导致 奇怪 的 结果 。 


SQLite Explain (解释 ) 


在 SQLite 语句 之 前 ， 可 以 使 用 "EXPLAIN" 关键 字 或 "EXPLAIN QUERY PLAN" 短 
语 ， 用 于 描述 表 的 细节 。 


如 果 省 略 了 EXPLAIN 关键 字 或 短语 ， 任 何 的 修改 都 会 引起 SQLite 语句 的 查询 行 
为 ， 并 返回 有 关 SQLite 语句 如 何 操 作 的 信息 。 


e 来 自 EXPLAIN 和 EXPLAIN QUERY PLAN 的 输出 只 用 于 交互 式 分 析 和 排除 故 
障 。 


。 输出 格式 的 细节 可 能 会 随 着 SQLite 版 本 的 不 同 而 有 所 变化 。 


e。 应 用 程序 不 应 该 使 用 EXPLAIN 或 EXPLAIN QUERY PLAN， 因 为 其 确切 的 行 
为 是 可 变 的 且 只 有 部 分 会 被 记录 。 


语法 
EXPLAIN 的 语法 如 下 : 


EXPLAIN [SQLite Query] 


EXPLAIN QUERY PLAN 的 语法 如 下 : 


EXPLAIN QUERY PLAN [SQLite Query] 


实例 


假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 

1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000 .0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


现在 ， 让 我 们 检查 SELECT 语句 中 的 Explain 使 用 : 


sqlite> EXPLAIN SELECT * FROM COMPANY WHERE Salary &gt= 20000; 


这 将 产生 以 下 结果 
addr opcode p1 p2 p3 
0 Goto 0 19 
1 Integer 0 0 
2 OpenRead 0 8 
3 SetNumColu © 5 
4 Rewind 0 17 
5 Column 0 4 
6 RealAffini 0 0 
7 Integer 20000 0 
8 Lt 357 16 collseq(BI 
9 Rowid 0 0 
10 Column 0 1 
11 Column 0 2 
12 Column 0 3 
13 Column 0 4 
14 RealAffini 0 0 
15 Callback 5 0 
16 Next 0 5 
17 Close 0 0 
18 Halt 0 0 
19 Transactio © 0 
20 VerifyCook 0 38 
21 Goto 0 1 
22 Noop 0 0 


现在 ， 让 我 们 检查 SELECT 语句 中 的 Explain Query Plan 使 用 : 


SQLite> EXPLAIN QUERY PLAN SELECT * FROM COMPANY WHERE Salary &gt= 





0 0 TABLE COMPANY 


SQLite Vacuum 


VACUUM 命 合 通 过 复制 主 数据 库 中 的 内 容 到 一 个 临时 数据 库 文件 ， 然 后 清空 主 数 
据 库 ， 并 从 副本 中 重新 载 人 原始 的 数据 库 文件 。 这 消除 了 空闲 页 ， 把 表 中 的 数据 排 
列 为 连续 的 ， 另 外 会 清理 数据 库 文件 结构 。 


如 果 表 中 没有 明确 的 整 型 主键 (INTEGER PRIMARY KEY) , VACUUM 命令 可 能 
会 改变 表 中 条 目的 行 ID (ROWID) > VACUUM 命令 只 适用 于 主 数据 库 ， 附 加 的 数 
据 库 文件 是 不 可 能 使 用 VACUUM AS. 


如 果 有 一 个 活动 的 事务 ，VACUUM 命令 就 会 失败 。VACUUM 命令 是 一 个 用 于 内 存 
数据 库 的 任何 操作 。 由 于 VACUUM 命令 从 头 开始 重新 创建 数据 库 文件 ， 所 以 
VACUUM 也 可 以 用 于 修改 许多 数据 库 特 定 的 配置 参数 。 


手动 VACUUM 
下 面 是 在 命令 提示 符 中 对 整个 数据 库 发 出 VACUUM 命令 的 语法 : 


$sqlite3 database_name "VACUUM;" 


您 也 可 以 在 SQLite 提示 符 中 运行 VACUUM， 如 下 所 示 : 


sqlite> VACUUM; 


您 也 可 以 在 特定 的 表 上 运行 VACUUM, BO RATA : 


sqlite> VACUUM table_name; 


Ea VACCUM (Auto-VACUUM) 


SQLite 的 Auto-VACUUM 与 VACUUM 不 大 一 样 ， 它 只 是 把 空闲 页 移 到 数据 库 末 
尾 ， 从 而 减 小 数据 库 大 小 。 通 过 这 样 做 ， 它 可 以 明显 地 把 数据 库 碎片 化 ， 而 
VACUUM 则 是 反 碎 片 化 。 所 以 Auto-VACUUM 只 会 让 数据 库 更 小 。 


在 SQLite 提示 符 中 ， 您 可 以 通过 下 面 的 编译 运行 ， 启用 /禁用 SQLite 的 Auto- 
VACUUM : 


sqlite> PRAGMA auto_vacuum = NONE; -- 0 means disable auto vacuum 
sqlite> PRAGMA auto_vacuum = INCREMENTAL; -- 1 means enable increr 
sqlite> PRAGMA auto_vacuum = FULL; -- 2 means enable full auto vac 
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您 可 以 从 命 命 提示 符 中 运行 下 面 的 命 合 来 检查 auto-vacuum 设置 : 


$sqlite3 database name "PRAGMA auto vacuum;" 


SQLite 日 期 & 时 间 


SQLite 支持 以 下 五 个 日 期 和 时 间 画 数 : 


函数 实例 
Aae MESING, DLYYYY-MM-DD 格式 返回 日 期 。 
modifiers...) 
e e 以 HH:MM:SS 格式 返回 时 间 。 
modifiers...) 
datetime(timestring, jy YYYY-MM-DD HH:MM:SS 格式 返回 。 
modifiers...) 
julianday(timestring, 这 将 返回 从 格林 尼 治 时 间 的 公元 前 4714 年 11 月 24 日 
modifiers...) 正午 算 起 的 天 数 。 
strftime(timestring, 这 将 根据 第 一 个 参数 指定 的 格式 字符 串 返 回 格式 化 的 日 
modifiers...) 期 。 具 体格 式 见 下 边 讲解 。 


上 述 五 个 日 期 和 时 间 画 数 把 时 间 字符 串 作 为 参数 。 时 间 字 符 串 后 跟 雾 个 或 多 个 
modifiers 修饰 符 。strftime() 画 数 也 可 以 把 格式 字符 捉 作 为 其 第 一 个 参数 。 下 面 将 
为 您 详细 讲解 不 同类 型 的 时 间 字 符 串 和 修饰 符 。 


时 间 字 符 串 


一 个 时 间 字 符 串 可 以 采用 下 面 任 何 一 种 格式 : 


时 间 字 符 串 实例 
YYYY-MM-DD 2010-12-30 
YYYY-MM-DD HH:MM 2010-12-30 12:10 
YYYY-MM-DD HH:MM:SS.SSS 2010-12-30 12:10:04.100 
MM-DD-YYYY HH:MM 30-12-2010 12:10 
HH:MM 12:10 
YYYY-MM-DDTHH:MM 2010-12-30 12:10 
HH:MM:SS 12:10:01 
YYYYMMDD HHMMSS 20101230 121001 
now 2013-05-07 


您 可 以 使 用 "T" 作为 分 隔日 期 和 时 间 的 文字 字符 。 


{E+ (Modifiers) 


时 间 字 符 串 后 边 可 跟着 震 个 或 多 个 的 修饰 符 ， 这 将 改变 有 上 述 五 个 画 数 返回 的 日 其 
和 /或 时 间 。 任 何 上 述 五 大 功能 返回 时 间 。 修 饰 符 应 从 左 到 右 使 用 ， 下 面 列 出 了 可 在 
SQLite 中 使 用 的 修饰 符 : 


e NNN days 
NNN hours 


NNN minutes 
e NNN.NNNN seconds 
NNN months 


NNN years 

e start of month 
e start of year 
e start of day 

e weekday N 

e unixepoch 


localtime 


e utc 


格式 化 


SQLite 提供 了 非常 方便 的 函数 strftime() 来 格式 化 任何 日 期 和 时 间 。 您 可 以 使 用 以 
下 的 替换 来 格式 化 日 期 和 时 间 : 


替换 描述 
%d 一 月 中 的 第 几 天 ，01-31 
%f 带 小 数 部 分 的 秒 ，SS.SSS 
%H 小 时 ，00-23 
%j 一 年 中 的 第 几 天 ，001-366 
%J 儒 略 日 数 ，DDDD.DDDD 
%m 月 ，00-12 
%M 分 ，00-59 
%s 从 1970-01-01 算 起 的 秒 数 
%S 秒 ，00-59 
%w 一 周 中 的 第 几 天 ，0-6 (0 is Sunday) 
%W 一 年 中 的 第 几 周 ，01-53 
%Y YY 
%% % symbol 
实例 


现在 让 我 们 使 用 SQLite 提示 符 尝 试 不 同 的 实例 。 下 面 是 计算 当前 日 期 : 


sqlite> SELECT date('now'); 
2013-05-07 


下 面 是 计算 当前 月 份 的 最 后 一 天 : 


sqlite> SELECT date('now', 'start of month','+1 month','-1 day'); 
2013-05-31 


下 面 是 计算 给 定 UNIX 时 间 惟 1092941466 的 日 期 和 时 间 : 


sqlite> SELECT datetime(1092941466, 'unixepoch'); 
2004-08-19 18:51:06 


下 面 是 计算 给 定 UNIX i HE 1092941466 相对 本 地 时 区 的 日 期 和 时 间 : 


sqlite> SELECT datetime(1092941466, '‘unixepoch', 'localtime'); 
2004-08-19 11:51:06 


下 面 是 计算 当前 的 UNIX at ja : 


sqlite> SELECT datetime(1092941466, 'unixepoch', 'localtime'); 
1367926057 


下 面 是 计算 美国 "独立 宣言 "签署 以 来 的 天 数 : 


sqlite> SELECT julianday('now') - julianday('1776-07-04'); 
86504 .4775830326 


下 面 是 计算 从 2004 年 某 一 特定 时 刻 以 来 的 秒 数 : 


sqlite> SELECT strftime('%s','now') - strftime('%s','2004-01-01 02 
295001572 


i 





下 面 是 计算 当年 10 月 的 第 一 个 星期 二 的 日 期 : 


sqlite> SELECT date('now','start of year','+9 months', 'weekday 2'), 
2013-10-01 
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下 面 是 计算 从 UNIX 纪元 算 起 的 以 秒 为 单位 的 时 间 (类 似 strftime(%s',now)， 不 
同 的 是 这 里 有 包括 小 数 部 分 ) : 


sqlite> SELECT (julianday('now') - 2440587.5)*86400.0; 
1367926077.12598 


在 UTC 与 本 地 时 间 值 之 间 进 行 转 换 ， 当 格式 化 日 期 时 ， 使 用 utc 或 localtime 修饰 
符 ， 如 下 所 示 : 


sqlite> SELECT time('12:00', ‘localtime' ); 
05:00:00 


sqlite> SELECT time('12:00', 


PiU ss yi 
19:00:00 


SQLite 常用 函数 


SQLite 有 许多 内 置 函 数 用 于 处 理 字符 串 或 数字 数据 。 下 面 列 出 了 一 些 有 用 的 


SQLite 内 置 画 数 ， 且 所 有 画 数 都 是 大 小 写 不 敏感 ， 这 意味 着 您 可 以 使 用 这 些 画 数 的 
小 写 形 式 或 大 写 形 式 或 混合 形式 。 欲 了 解 更 多 详情 ， 请 查看 SQLite 的 官方 文档 : 
SQLite SQLite COUNT 聚集 函数 是 用 来 计算 一 个 数据 库 表 中 的 行 

COUNT 函数 数 。 
Spr MAX SQLite MAX 聚合 画 数 允许 我 们 选择 某 列 的 最 大 值 。 
TA MIN E SQLite MIN 聚合 函数 允许 我 们 选择 某 列 的 最 小 值 。 
SQLite AVG SQLite AVG 聚合 函数 计算 某 列 的 平均 值 。 
PRB 
al SUM SQLite SUM 聚合 画 数 允 许 为 一 个 数值 列 计算 总 和 。 
SQLite SQLite RANDOM 画 数 返 回 一 个 介 于 
RANDOM i -9223372036854775808 和 +9223372036854775807 之 间 
数 的 伪 随 机 整数 。 
SQLite ABS 
naan SQLite ABS 函数 返回 数值 参数 的 绝对 值 。 
SQLite 
UPPER Hit SQLite UPPER 函数 把 字符 串 转 换 为 大 写字 母 。 
SQLite SQLite LOWER 画 数 把 字符 串 转 换 为 小 写字 母 。 
LOWER fz 
SQLite 
LENGTH ma SQLite LENGTH 画 数 返 回 字符 串 的 长 度 。 
SQLite 
sqlite_version SQLite sqlite_version 函数 返回 SQLite 库 的 版 本 。 
PER 

在 我 们 开始 讲解 这 些 函 数 实 例 之 前 ， 先 假设 COMPANY 表 有 以 下 记录 : 


ID NAME AGE ADDRESS SALARY 


1 Paul 32 California 20000.0 
2 Allen 25 Texas 15000.0 
3 Teddy 23 Norway 20000.0 
4 Mark 25 Rich-Mond 65000.0 
5 David 27 Texas 85000.0 
6 Kim 22 South-Hall 45000.0 
7 James 24 Houston 10000.0 


SQLite COUNT 函数 

SQLite COUNT B# EMER Kit S—TREERAMTM. FRELO: 
sqlite> SELECT count(*) FROM COMPANY; 

ERMA SQLite SQL 语句 将 产生 以 下 结果 : 


count(*) 


SQLite MAX 函数 

SQLite MAX 聚合 西数 允许 我 们 选择 某 列 的 最 大 值 。 下 面 是 实例 : 
sqlite> SELECT max(salary) FROM COMPANY; 

上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


max(salary) 


85000 .0 


SQLite MIN 函数 
SQLite MIN RA WRARHRiNBERIINRI GE. TAERA: 


sqlite> SELECT min(salary) FROM COMPANY; 


上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


min(salary) 


10000.0 


SQLite AVG 函数 

SQLite AVG REKHA RIJEI. FREES: 
sqlite> SELECT avg(salary) FROM COMPANY; 

上 面 的 SQLite SQL 话 句 将 产生 以 下 结果 : 


avg(salary) 


37142 .8571428572 


SQLite SUM 函数 

SQLite SUM 聚合 西数 允许 为 一 个 数值 列 计算 总 和 。 下 面 是 实例 : 
sqlite> SELECT sum(salary) FROM COMPANY; 

上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


sum(salary) 


260000.0 


SQLite RANDOM 函数 


SQLite RANDOM 函数 返回 一 个 介 于 -9223372036854775808 和 
+9223372036854775807 之 间 的 伪 随 机 整数 。 下 面 是 实例 : 


sqlite> SELECT random() AS Random; 


上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


Random 


5876796417670984050 


SQLite ABS 函数 
SQLite ABS 函数 返回 数值 参数 的 绝对 值 。 下 面 是 实例 : 

sqlite> SELECT abs(5), abs(-15), abs(NULL), abs(0), abs("ABC"); 
上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


abs(5) abs(-15) abs(NULL) abs(0) abs ("ABC") 


SQLite UPPER iz 

SQLite UPPER 函数 把 字符 串 转 换 为 大 写字 母 。 下 面 是 实例 : 
sqlite> SELECT upper(name) FROM COMPANY; 

上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


upper (name) 


SQLite LOWER HX 
SQLite LOWER 函数 把 字符 串 转 换 为 小 写字 母 。 下 面 是 实例 : 


sqlite> SELECT lower(name) FROM COMPANY; 


上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


lower (name) 


SQLite LENGTH 2x 

SQLite LENGTH 画 数 返 回 字符 串 的 长 度 。 下 面 是 实例 : 
sqlite> SELECT name, length(name) FROM COMPANY; 

上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


NAME length(name) 


SQLite sqlite _version 函数 

SQLite sqlite_version WŽUGRE| SQLite 库 的 版 本 。 下 面 是 实例 : 
sqlite> SELECT sqlite version() AS 'SQLite Version'; 

上 面 的 SQLite SQL 语句 将 产生 以 下 结果 : 


SQLite Version 


SQLite 接口 


SQLite - C/C++ 
安 委 


在 C/C++ 程序 中 使 用 SQLite 之 前 ， 我 们 需要 确保 机 器 上 已 经 有 SQLite 库 。 可 以 
查看 SQLite 安装 章节 了 解 安 装 过 程 。 


C/C++ 接口 API 


以 下 是 重要 的 C&C++ / SQLite 接口 程序 ， 可 以 满足 您 在 C/C++ 程序 中 使 用 
SQLite 数据 库 的 需求 。 如 果 您 需要 了 解 更 多 细节 ， 请 查看 SQLite 官方 文档 。 


API 描述 


该 例 程 打开 一 个 指向 SQLite 数据 库 文件 的 连接 ， 返 

回 一 个 用 于 其 他 SQLite 程序 的 数据 库 连 接 对 象 。 

如 果 filename 参数 是 NULL 或 :memory:"， 那 么 
sqlite3_open(const sqlite3_open() 将 会 在 RAM 中 创建 一 个 内 存 数据 


char *filename， 库 ， 这 只 会 在 session 的 有 效 时 间 内 持续 。 如 果 文 
sqlite3 “ppDb) 件 名 filename 不 为 NULL， 那 么 sqlite3_open() 将 


使 用 这 个 参数 值 党 试 打开 数据 库 文 件 。 如 果 该 名 称 
的 文件 不 存在 ，sqlite3_open() 将 创建 一 个 新 的 命名 
为 该 名 称 的 数据 库 文件 并 打开 。 


该 例 程 提供 了 一 个 执行 SQL 命令 的 快捷 方式 ，SQL 

命令 由 sql 参数 提供 ， 可 以 由 多 个 SQL 命令 组 成 。 
sqlite3_exec(sqlite3*, 在 这 里 ， 第 一 个 参数 sqlite3 是 打开 的 数据 库 对 
const char *sql, &, sqlite_callback 是 一 个 回调 ，data 作为 其 第 一 
sqlite_callback, void TSA, errmsg 将 被 返回 用 来 获取 程序 生成 的 任何 
*data, char **errmsg) 错误 。 sqlite3_exec() 程序 解析 并 执行 由 sql 参数 所 

给 的 每 个 命令， 直到 字符 串 结束 或 者 遇 到 错误 为 

ies 


该 例 程 关 闭 之 前 调用 sqlite3_open() 打开 的 数据 库 
连接 。 所 有 和 与 连接 相关 的 语句 都 应 在 连接 关闭 之 前 
完成 。 如 果 还 有 查询 没有 完成 ，sqlite3_close() 将 
返回 SQLITE_BUSY 禁止 关闭 的 错误 消息 。 


sqlite3_close(sqlite3*) 


连接 数据 库 


下 面 的 C 代码 段 显示 了 如 何 连 接 到 一 个 现 有 的 数据 库 。 如 果 数 据 库 不 存在 ， 那 么 它 
就 会 被 创建 ， 最 后 将 返回 一 个 数据 库 对 象 。 


#include <stdio.h> 
#include <sqlite3.h> 


int main(int argc, char* argv[]) 


{ 
sqlite3 *db; 
char *zErrMsg = 0; 
int rc; 
rc = sqlite3_open("test.db", &db); 
i re ){ 
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg( 
exit(d); 
selse{ 
fprintf(stderr, "Opened database successfully\n"); 
} 
sqlite3_close(db); 
} 





现在 ， 让 我 们 来 编译 和 运行 上 面 的 程序 ， 在 当前 目录 中 创建 我 们 的 数据 库 
test.db。 您 可 以 根据 需要 改变 路 径 。 


$gcc test.c -1 sqlite3 
$./a.out 
Opened database successfully 


如 果 要 使 用 C++ 源 代 码 ， 可 以 按照 下 列 所 示 编 译 代 码 : 


$g++ test.c -l sqlite3 


在 这 里 ， 把 我 们 的 程序 链接 上 sqlite3 库 ， 以 便 向 C 程序 提供 必要 的 函数 。 这 将 在 
您 的 目录 下 创建 一 个 数据 库 文 件 test.db， 您 将 得 到 如 下 结果 : 


-rwxr-xr-x. 1 root root 7383 May 8 02:06 a.out 
-rw-r--r--. 1 root root 323 May 8 02:05 test.c 
-rw-r--r--. 1 root root © May 8 02:06 test.db 


创建 表 


下 面 的 C 代码 段 将 用 于 在 先前 创建 的 数据 库 中 创建 一 个 表 : 


#include <stdio.h> 
#include <stdlib.h> 
#include <sqlite3.h> 


static int callback(void *NotUsed, int argc, char **argv, char **a; 


} 


int i; 
for(i=0; i&lt;argc; i++){ 
printf ("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL" 


} 
printf("\n"); 
return 0; 


int main(int argc, char* argv[]) 


{ 


sqlite3 *db; 

char *zErrMsg = 0; 
int rc; 

char *sql; 


/* Open database */ 

rc = sqlite3_open("test.db", &db); 

if( rc ){ 
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg( 
exit(0); 

selse{ 
fprintf(stdout, "Opened database successfully\n"); 

} 


/* Create SQL statement */ 
sql = "CREATE TABLE COMPANY(" \ 


"ID INT PRIMARY KEY NOT NULL," \ 
"NAME TEXT NOT NULL," \ 
"AGE INT NOT NULL," \ 
"ADDRESS CHAR(50)," \ 

"SALARY REAL );"; 


/* Execute SQL statement */ 
rc = sqlite3_exec(db, sql, callback, 0, &zErrMsg); 
if( rc != SQLITE_OK ){ 
fprintf(stderr, "SQL error: %s\n", zErrMsg); 
sqlite3_free(zErrMsg); 
selse{ 
fprintf(stdout, "Table created successfully\n"); 


} 
sqlite3 close(db); 
return 0; 








上 述 程序 编译 和 执行 时 ， 它 会 在 test.db 文件 中 创建 COMPANY 表 ， 最 终 文件 列表 
如 下 所 示 : 


-rwxr-xr-x. 1 root root 9567 May 8 02:31 a.out 
-rw-r--r--. 1 root root 1207 May 8 02:31 test.c 
-rw-r--r--. 1 root root 3072 May 8 02:31 test.db 


INSERT 操作 


下 面 的 C 代码 段 显示 了 如 何在 上 面 创建 的 COMPANY 表 中 创建 记录 : 


#include <stdio.h> 
#include <stdlib.h> 
#include <sqlite3.h> 


static int callback(void *NotUsed, int argc, char **argv, char **a; 


int i; 
for(i=0; i&lt;argc; i++){ 
printf ("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL" 


} 
printf("\n"); 
return 0; 


} 


int main(int argc, char* argv[]) 
{ 

sqlite3 *db; 

char *zErrMsg = 0; 

int rc; 

char *sql; 


/* Open database */ 

rc = sqlite3_open("test.db", &db); 

if( rc ){ 
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(( 
exit(0); 

}else{ 
fprintf(stderr, "Opened database successfully\n"); 

} 


/* Create SQL statement */ 

sql = "INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) " \ 
"VALUES (1, 'Paul', 32, 'California', 20000.00 ); " \ 
"INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) " \ 
"VALUES (2, 'Allen', 25, 'Texas', 15000.00 ); " \ 
"INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY)" \ 
"VALUES (3, 'Teddy', 23, 'Norway', 20000.00 );" \ 
"INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY)" \ 
"VALUES (4, 'Mark', 25, 'Rich-Mond ', 65000.00 );"; 


/* Execute SQL statement */ 
rc = sqlite3_exec(db, sql, callback, 0, &zErrMsg); 
if( rc != SQLITE OK ){ 
fprintf(stderr, "SQL error: %s\n", zErrMsg); 
sqlite3_free(zErrMsg); 
}else{ 
fprintf(stdout, "Records created successfully\n"); 


} 
sqlite3_close(db); 
return 0; 








上 述 程序 编译 和 执行 时 ， 它 会 在 COMPANY 表 中 创建 给 定 记 录 ， 并 会 显示 以 下 两 
行 : 


Opened database successfully 
Records created successfully 


SELECT 操作 

在 我 们 开始 讲解 获取 记录 的 实例 之 前 ， 让 我 们 先 了 解 下 回调 函数 的 一 些 细节 ， 这 将 
在 我 们 的 实例 使 用 到 。 这 个 回调 提供 了 一 个 从 SELECT 语句 获得 结果 的 方式 。 它 
声明 如 下 : 


typedef int (*sqlite3_callback) ( 


void? ; /* Data provided in the 4th argument of sqlite3_exec() *, 
int, /* The number of columns in row */ 

char**, /* An array of strings representing fields in the row */ 
char ** /* An array of strings representing column names */ 





如 果 上 面 的 回调 在 sqlite_exec() 程序 中 作为 第 三 个 参数 ， 那 么 SQLite 将 为 SQL 参 
数 内 执行 的 每 个 SELECT 语句 中 处 理 的 每 个 记录 调用 这 个 回调 函数 。 


FAN C 代码 段 显 示 了 如 何 从 前 面 创 建 的 COMPANY 表 中 获取 并 显示 记录 : 


#include <stdio.h> 
#include <stdlib.h> 
#include <sqlite3.h> 


static int callback(void *data, int argc, char **argv, char **azCo- 
int i; 
fprintf(stderr, "%s: ", (const char*)data); 
for(i=0; i&lt;argc; i++){ 
printf ("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL" 


} 
printf ("\n"); 


return 0; 
} 
int main(int argc, char* argv[]) 
{ 
sqlite3 *db; 
char *zErrMsg = 0; 
IMENE, 
char *sql; 
const char* data = "Callback function called"; 
/* Open database */ 
rc = sqlite3_open("test.db", &db); 
if( rc ){ 
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(( 
exit(0); 
}else{ 
fprintf(stderr, "Opened database successfully\n"); 
} 
/* Create SQL statement */ 
sql = "SELECT * from COMPANY"; 
/* Execute SQL statement */ 
rc = sqlite3_exec(db, sql, callback, (void*)data, &zErrMsg); 
if( rc != SQLITE OK ){ 
fprintf(stderr, "SQL error: %s\n", ZErrMsg); 
sqlite3_free(zErrMsg); 
selse{ 
fprintf(stdout, "Operation done successfully\n"); 
} 
sqlite3 close(db); 
return 0; 
} 





上 述 程 序 编译 和 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
Callback function called: ID 
NAME = Paul 

AGE = 32 

ADDRESS = California 

SALARY = 20000.0 


I 
m 


Callback function called: ID 
NAME = Allen 

AGE = 25 

ADDRESS = Texas 

SALARY = 15000.0 


ll 
N 


Callback function called: ID 
NAME = Teddy 

AGE = 23 

ADDRESS = Norway 

SALARY = 20000.0 


ll 
w 


Callback function called: ID 
NAME = Mark 

AGE = 25 

ADDRESS = Rich-Mond 

SALARY = 65000.0 


ll 
小 


Operation done successfully 


UPDATE 操作 


TAH C 代码 段 显示 了 如 何 使 用 UPDATE 语句 来 更 新 任何 记录 ， 然 后 从 
COMPANY 表 中 获取 并 显示 更 新 的 记录 : 


#include <stdio.h> 
#include <stdlib.h> 
#include <sqlite3.h> 


static int callback(void *data, int argc, char **argv, char **azCo- 
int i; 
fprintf(stderr, "%s: ", (const char*)data); 
for(i=0; i&lt;argc; i++){ 


printf ("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL" 
} 
printf("\n"); 
return 0; 
} 
int main(int argc, char* argv[]) 
{ 
sqlite3 *db; 
char *zErrMsg = 0; 
IMENE, 
char *sql; 
const char* data = "Callback function called"; 
/* Open database */ 
rc = sqlite3_open("test.db", &db); 
af( Fe DH 
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(< 
exit(0); 
selse{ 
fprintf(stderr, "Opened database successfully\n"); 
} 
/* Create merged SQL statement */ 
sql = "UPDATE COMPANY set SALARY = 25000.00 where ID=1; " \ 
"SELECT * from COMPANY"; 
/* Execute SQL statement */ 
rc = sqlite3_exec(db, sql, callback, (void*)data, &zErrMsg); 
if( rc != SQLITE_OK ){ 
fprintf(stderr, "SQL error: %s\n", ZErrMsg); 
sqlite3_free(zErrMsg); 
selse{ 
fprintf(stdout, "Operation done successfully\n"); 
} 
sqlite3_close(db); 
return 0; 
} 





上 述 程 序 编译 和 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
Callback function called: ID 
NAME = Paul 

AGE = 32 

ADDRESS = California 

SALARY = 25000.0 


I 
m 


Callback function called: ID 
NAME = Allen 

AGE = 25 

ADDRESS = Texas 

SALARY = 15000.0 


ll 
N 


Callback function called: ID 
NAME = Teddy 

AGE = 23 

ADDRESS = Norway 

SALARY = 20000.0 


ll 
w 


Callback function called: ID 
NAME = Mark 

AGE = 25 

ADDRESS = Rich-Mond 

SALARY = 65000.0 


ll 
小 


Operation done successfully 


DELETE 操作 


TAH C 代码 段 显示 了 如 何 使 用 DELETE 语句 删除 任何 记录 ， 然 后 从 COMPANY 
表 中 获取 并 显示 剩余 的 记录 : 


#include <stdio.h> 
#include <stdlib.h> 
#include <sqlite3.h> 


static int callback(void *data, int argc, char **argv, char **azCo- 
int i; 
fprintf(stderr, "%s: ", (const char*)data); 
for(i=0; i<argc; i++){ 


printf ("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL" 
} 
printf("\n"); 
return 0; 
} 
int main(int argc, char* argv[]) 
{ 
sqlite3 *db; 
char *zErrMsg = 0; 
IMENE, 
char *sql; 
const char* data = "Callback function called"; 
/* Open database */ 
rc = sqlite3_open("test.db", &db); 
af( Fe DH 
fprintf(stderr, "Can't open database: %s\n", sqlite3_errmsg(< 
exit(0); 
selse{ 
fprintf(stderr, "Opened database successfully\n"); 
} 
/* Create merged SQL statement */ 
sql = "DELETE from COMPANY where ID=2; " \ 
"SELECT * from COMPANY"; 
/* Execute SQL statement */ 
rc = sqlite3_exec(db, sql, callback, (void*)data, &zErrMsg); 
if( rc != SQLITE_OK ){ 
fprintf(stderr, "SQL error: %s\n", ZErrMsg); 
sqlite3_free(zErrMsg); 
selse{ 
fprintf(stdout, "Operation done successfully\n"); 
} 
sqlite3_close(db); 
return 0; 
} 





上 述 程 序 编译 和 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
Callback function called: ID 
NAME = Paul 

AGE = 32 

ADDRESS = California 

SALARY = 20000.0 


I 
m 


Callback function called: ID 
NAME = Teddy 

AGE = 23 

ADDRESS = Norway 

SALARY = 20000.0 


ll 
wo 


Callback function called: ID 
NAME = Mark 

AGE = 25 

ADDRESS = Rich-Mond 

SALARY = 65000.0 


ll 
小 


Operation done successfully 


SQLite - Java 
安装 


在 Java 程序 中 使 用 SQLite 之 前 ， 我 们 需要 确保 机 器 上 已 经 有 SQLite JDBC 
Driver 驱动 程序 和 Java。 可 以 查看 Java 教程 了 解 如 何在 计算 机 上 安装 Java, I 
在 ， 我 们 来 看 看 如 何在 机 器 上 安装 SQLite JDBC 驱动 程序 。 


。 从 sqlite-jdbc 库 下 载 sqlite-jdbc-(VERSION).jar 的 最 新 版 本 。 


o 在 您 的 class 路 径 中 添加 下 载 的 jar 文件 sqlite-jqbc-(VERSION).jar， 或 者 在 - 
classpath 选项 中 使 用 它 ， 这 将 在 后 面 的 实例 中 进行 讲解 。 


在 学 习 下 面部 分 的 知识 之 前 ， 您 必须 对 Java JDBC 概念 有 初步 了 解 。 如 果 您 还 未 
了 解 相 关 知 识 ， 那 么 建议 您 可 以 先 花 半 个 小 时 学 习 下 JDBC 教程 相关 知识 ， 这 将 有 
助 于 您 学 习 接 下 来 讲解 的 知识 。 


连接 数据 库 


TFAM Java 程序 显示 了 如 何 连接 到 一 个 现 有 的 数据 库 。 如 果 数 据 库 不 存在 ， 那 么 
它 就 会 被 创建 ， 最 后 将 返回 一 个 数据 库 对 象 。 


import java.sql.*; 
public class SQLiteJDBC 


public static void main( String args[] ) 
{ 
Connection c = null; 
try { 
Class.forName("org.sqlite. JDBC"); 
c = DriverManager.getConnection("jdbc:sqlite:test.db"); 
} catch ( Exceptione ) { 
System.err.printlin( e.getClass().getName() + ": " + e.getMes: 
System.exit(0); 
} 


System.out.println("Opened database successfully"); 





现在 ， 让 我 们 来 编译 和 运行 上 面 的 程序 ， 在 当前 目录 中 创建 我 们 的 数据 库 
test.db。 您 可 以 根据 需要 改变 路 径 。 我 们 假设 当前 路 径 下 可 用 的 JDBC 驱动 程序 
的 版 本 是 sqlite-jdbc-3.7.2.jar。 


$javac SQLiteJDBC. java 
$java -classpath ".:sqlite-jdbc-3.7.2.jar" SQLiteJDBC 
Open database successfully 


如 果 您 想 要 使 用 Windows 机 器 ， 可 以 按照 下 列 所 示 编 译 和 运行 您 的 代码 : 


$javac SQLiteJDBC. java 
$java -classpath ".;sqlite-jdbc-3.7.2.jar" SQLiteJDBC 
Opened database successfully 


创建 表 
下 面 的 Java 程序 将 用 于 在 先前 创建 的 数据 库 中 创建 一 个 表 : 


import java.sql.*; 


public class SQLiteJDBC 


{ 
public static void main( String args[] ) 
{ 
Connection c = null; 
Statement stmt = null; 
try { 
Class.forName("org.sqlite.JDBC"); 
c = DriverManager.getConnection("jdbc:sqlite:test.db"); 
System.out.println("Opened database successfully"); 
stmt = c.createStatement(); 
String sql = "CREATE TABLE COMPANY " + 
"(ID INT PRIMARY KEY NOT NULL," + 
" NAME TEXT NOT NULL, " + 
" AGE INT NOT NULL, " + 
" ADDRESS CHAR(50), " + 
" SALARY REAL)"; 
stmt.executeUpdate(sql); 
stmt.close(); 
c.close(); 
} catch ( Exception e ) { 
System.err.printin( e.getClass().getName() + ": " + e.getMes: 
System.exit(0); 
} 
System.out.println("Table created successfully"); 
} 


} 
«| == 











上 述 程序 编译 和 执行 时 ， 它 会 在 test.db 中 创建 COMPANY 表 ， 最 终 文件 列表 如 下 
所 示 : 





-rw-r--r--. 1 root root 3201128 Jan 22 19:04 sqlite-jdbc-3.7.2.jar 
-rw-r--r--. 1 root root 1506 May 8 05:43 SQLiteJDBC.class 
-rw-r--r--. 1 root root 832 May 8 05:42 SQLiteJDBC.java 
-rw-r--r--. 1 root root 3072 May 8 05:43 test.db 
‘| EE Be > | 
[=] 
INSERT 操作 


下 面 的 Java 代码 显示 了 如 何在 上 面 创建 的 COMPANY 表 中 创建 记录 : 


import java.sql.*; 


public class SQLiteJDBC 


{ 
public static void main( String args[] ) 
{ 
Connection c = null; 
Statement stmt = null; 
try { 
Class. forName("org.sqlite. JDBC"); 
c = DriverManager.getConnection("jdbc:sqlite:test.db"); 
c.setAutoCommit (false); 
System.out.println("Opened database successfully"); 
stmt = c.createStatement(); 
String sql = "INSERT INTO COMPANY (ID,NAME, AGE, ADDRESS, SALAR* 
"VALUES (1, 'Paul', 32, 'California', 20000.00 ` 
stmt.executeUpdate(sql); 
sql = "INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) " + 
"VALUES (2, ‘Allen', 25, 'Texas', 15000.00 );"; 
stmt.executeUpdate(sql); 
sql = "INSERT INTO COMPANY (1ID,NAME,AGE,ADDRESS, SALARY) " + 
"VALUES (3, 'Teddy', 23, 'Norway', 20000.00 );"; 
stmt.executeUpdate(sql); 
sql = "INSERT INTO COMPANY (1ID,NAME,AGE,ADDRESS, SALARY) " + 
"VALUES (4, 'Mark', 25, 'Rich-Mond ', 65000.00 );"; 
stmt.executeUpdate(sql); 
stmt.close(); 
c.commit(); 
c.close(); 
} catch ( Exception e ) { 
System.err.printin( e.getClass().getName() + ": " + e.getMes: 
System.exit(0); 
} 
System.out.println("Records created successfully"); 
} 
} 





上 述 程序 编译 和 执行 时 ， 它 会 在 COMPANY 表 中 创建 给 定 记 录 ， 并 会 显示 以 下 两 
行 : 


Opened database successfully 
Records created successfully 


SELECT 操作 
下 面 的 Java 程序 显示 了 如 何 从 前 面 创建 的 COMPANY 表 中 获取 并 显示 记录 : 


import java.sql.*; 


public class SQLiteJDBC 
{ 
public static void main( String args[] ) 
{ 
Connection c = null; 
Statement stmt = null; 
try { 
Class.forName("org.sqlite.JDBC"); 
c = DriverManager.getConnection("jdbc:sqlite:test.db"); 
c.setAutoCommit (false); 
System.out.println("Opened database successfully"); 


stmt = c.createStatement(); 
ResultSet rs = stmt.executeQuery( "SELECT * FROM COMPANY;" ), 
while ( rs.next() ) { 
int id = rs.getInt("id"); 
String name = rs.getString("name"); 
int age = rs.getInt("age"); 
String address = rs.getString("address"); 
float salary = rs.getFloat("salary"); 
System.out.printin( "ID = " + id ); 
System.out.println( "NAME = " + name ); 
System.out.printin( "AGE = " + age ); 
System.out.println( "ADDRESS = " + address ); 
System.out.printin( "SALARY = " + salary ); 
System.out.printin(); 


rs.close(); 
stmt.close(); 
c.close(); 
} catch ( Exceptione ) { 
System.err.printin( e.getClass().getName() + ": " + e.getMes: 
System.exit(0); 
} 


System.out.println("Operation done successfully"); 





上 述 程序 编译 和 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 


It[D) = a 
NAME = Paul 
AGE = 32 


ADDRESS = California 
SALARY = 20000.0 


TD y=" 2 
NAME = Allen 
AGE = 25 


ADDRESS = Texas 
SALARY = 15000.0 


ID = 3 
NAME = Teddy 
AGE = 23 


ADDRESS = Norway 
SALARY = 20000.0 


ID = 4 
NAME = Mark 
AGE = 25 


ADDRESS = Rich-Mond 
SALARY = 65000.0 


Operation done successfully 


UPDATE 操作 


Fl Java 代码 显示 了 如 何 使 用 UPDATE 语句 来 更 新 任何 记录 ， 然 后 从 
COMPANY 表 中 获取 并 显示 更 新 的 记录 : 


import java.sql.*; 


public class SQLiteJDBC 
{ 
public static void main( String args[] ) 
{ 
Connection c = null; 
Statement stmt = null; 
try { 
Class. forName("org.sqlite. JDBC"); 
c = DriverManager.getConnection("jdbc:sqlite:test.db"); 
c.setAutoCommit (false); 
System.out.println("Opened database successfully"); 


stmt = c.createStatement(); 

String sql = "UPDATE COMPANY set SALARY = 25000.00 where ID=: 
stmt.executeUpdate(sql); 

c.commit(); 


ResultSet rs = stmt.executeQuery( "SELECT * FROM COMPANY;" ), 
while ( rs.next() ) { 
int id = rs.getInt("id"); 
String name = rs.getString("name"); 
int age = rs.getInt("age"); 
String address = rs.getString("address"); 
float salary = rs.getFloat("salary"); 
System.out.printin( "ID = " + id ); 
System.out.println( "NAME = " + name ); 
System.out.println( "AGE = " + age ); 
System.out.printin( "ADDRESS = " + address ); 
System.out.println( "SALARY = " + salary ); 
System.out.printiln(); 


rs.close(); 
stmt.close(); 
c.close(); 
} catch ( Exception e ) { 
System.err.printin( e.getClass().getName() + ": " + e.getMes: 
System.exit(0); 
} 


System.out.println("Operation done successfully"); 





上 述 程序 编译 和 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 


It[D) = a 
NAME = Paul 
AGE = 32 


ADDRESS = California 
SALARY = 25000.0 


TD =" 2 
NAME = Allen 
AGE = 25 


ADDRESS = Texas 
SALARY = 15000.0 


ID = 3 
NAME = Teddy 
AGE = 23 


ADDRESS = Norway 
SALARY = 20000.0 


ID = 4 
NAME = Mark 
AGE = 25 


ADDRESS = Rich-Mond 
SALARY = 65000.0 


Operation done successfully 


DELETE 操作 


下 面 的 Java 代码 显示 了 如 何 使 用 DELETE 语句 删除 任何 记录 ， 然 后 从 COMPANY 
表 中 获取 并 显示 剩余 的 记录 : 


import java.sql.*; 


public class SQLiteJDBC 
{ 
public static void main( String args[] ) 
{ 
Connection c = null; 
Statement stmt = null; 
try { 
Class. forName("org.sqlite. JDBC"); 
c = DriverManager.getConnection("jdbc:sqlite:test.db"); 
c.setAutoCommit (false); 
System.out.println("Opened database successfully"); 


stmt = c.createStatement(); 

String sql = "DELETE from COMPANY where ID=2;"; 
stmt.executeUpdate(sql); 

c.commit(); 


ResultSet rs = stmt.executeQuery( "SELECT * FROM COMPANY;" ), 
while ( rs.next() ) { 
int id = rs.getInt("id"); 
String name = rs.getString("name"); 
int age = rs.getInt("age"); 
String address = rs.getString("address"); 
float salary = rs.getFloat("salary"); 
System.out.printin( "ID = " + id ); 
System.out.println( "NAME = " + name ); 
System.out.println( "AGE = " + age ); 
System.out.printin( "ADDRESS = " + address ); 
System.out.println( "SALARY = " + salary ); 
System.out.printiln(); 


rs.close(); 
stmt.close(); 
c.close(); 
} catch ( Exception e ) { 
System.err.printlin( e.getClass().getName() + ": " + e.getMes: 
System.exit(0); 
} 


System.out.println("Operation done successfully"); 





上 述 程序 编译 和 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
ID = 1 

NAME = Paul 

AGE = 32 

ADDRESS = California 

SALARY = 25000.0 


ID = 3 
NAME = Teddy 
AGE = 23 


ADDRESS = Norway 
SALARY = 20000.0 


ID = 4 
NAME = Mark 
AGE = 25 


ADDRESS = Rich-Mond 
SALARY = 65000.0 


Operation done successfully 


SQLite - PHP 
RR 


自 PHP 5.3.0 起 默认 启用 SQLite3 扩展 。 可 以 在 编译 时 使 用 --without-sqlite3 4 
用 SQLite3 扩展 。 


Windows 用 户 必须 启用 php_sqlite3.dll 才能 使 用 该 扩展 。 自 PHP 5.3.0 起 ， 这 个 
DLL 被 包含 在 PHP 的 Windows 分 发 版 中 。 


如 需 了 解 详 细 的 安装 指导 ， 建 议 查 看 我 们 的 PHP 教程 和 它 的 官方 网 站 。 
PHP 接口 API 


以 下 是 重要 的 PHP 程序 ， 可 以 满足 您 在 PHP 程序 中 使 用 SQLite 数据 库 的 需求 。 
如 果 您 需要 了 解 更 多 细节 ， 请 查看 PHP 官方 文档 。 


API 


public void 
SQLite3::open ( 
filename, flags, 
encryption_key ) 


public bool 
SQLite3::exec ( string 
$query ) 


public SQLite3Result 
SQLite3::query ( string 
$query ) 


public int 
SQLite3::lastErrorCode 
( void ) 


public string 
SQLite3::lastErrorMsg 
( void ) 


public int 
SQLite3::changes ( 
void ) 


public bool 
SQLite3::close ( void ) 


public string 
SQLite3::escapeString 
( string $value ) 


描述 


打开 一 个 SQLite 3 数据 库 。 如 果 构 建 包 括 加 密 ， 
那么 它 将 尝试 使 用 的 密 钥 。 如 果 文 件 名 filename 
赋值 为 '"memory:"， 那 么 SQLite3::open() 将 会 在 
RAM 中 创建 一 个 内 存 数据 库 ， 这 只 会 在 session 
的 有 效 时 间 内 持续 。 MRE filename 为 实际 
的 设备 文件 名 称 ， 那 么 SQLite3::open() 将 使 用 这 
个 参数 值 党 试 打开 数据 库 文 件 。 如 果 该 名 称 的 文件 
不 存在 ， 那 么 将 创建 一 个 新 的 命名 为 该 名 称 的 数据 
库 文 件 。 可 选 的 flags 用 于 判断 是 否 打开 SQLite 
数据 库 。 默 认 情 况 下 ， 当 使 用 
SQLITE3_OPEN_READWRITE | 
SQLITE3_OPEN_CREATE 时 打开 。 


该 例 程 提供 了 一 个 执行 SQL 命令 的 快捷 方式 ， 
SQL 命令 由 sql 参数 提供 ， 可 以 由 多 个 SQL HR 
组 成 。 该 程序 用 于 对 给 定 的 数据 库 执行 一 个 无 结果 
的 查询 。 


该 例 程 执行 一 个 SQL 查询 ， 如 果 查 询 到 返回 结果 
则 返回 一 个 SQLite3Result 对 象 。 


该 例 程 返回 最 近 一 次 失败 的 SQLite 请 求 的 数值 结 
RK. 


该 例 程 返回 最 近 一 次 失败 的 SQLite 请 求 的 英语 文 


该 例 程 返 回 最 近 一 次 的 SQL 语句 更 新 或 插 和 人 或 删 
除 的 数据 库 行 数 。 


该 例 程 关 闭 之 前 调用 SQLite3::open() 打开 的 数据 
库 连 接 。 


该 例 程 返 回 一 个 字符 串 ， 在 SQL 语句 中 ， 出 于 安 
全 考虑 ， 该 字符 串 已 被 正确 地 转 义 。 


连接 数据 库 


下 面 的 PHP 代码 显示 了 如 何 连接 到 一 个 现 有 的 数据 库 。 如 果 数 据 库 不 存在 ， 那 么 
它 就 会 被 创建 ， 最 后 将 返回 一 个 数据 库 对 象 。 


<?php 
class MyDB extends SQLite3 
{ 


function __construct() 
$this->open('test.db'); 
} 
} 
$db = new MyDB(); 
if(!$db){ 
echo $db->lastErrorMsg(); 


} else { 
echo "Opened database successfully\n"; 
} 


?> 


现在 ， 让 我 们 来 运行 上 面 的 程序 ， 在 当前 目录 中 创建 我 们 的 数据 库 test.db。 您 可 
以 根据 需要 改变 路 径 。 如 果 数 据 库 成 功 创建 ， 那 么 会 显示 下 面 所 示 的 消息 : 


Open database successfully 


创建 表 
下 面 的 PHP 代码 段 将 用 于 在 先前 创建 的 数据 库 中 创建 一 个 表 : 


<?php 
class MyDB extends SQLite3 
{ 


function __construct() 


$this->open('test.db'); 
} 


} 
$db = new MyDB(); 


if(!$db){ 

echo $db->lastErrorMsg(); 
} else { 

echo "Opened database successfully\n"; 
} 


$sql =<<<EOF 
CREATE TABLE COMPANY 


(ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 
SALARY REAL); 
EOF; 

$ret = $db->exec($sql); 

if(!$ret){ 
echo $db->lastErrorMsg(); 

} else { 


echo "Table created successfully\n"; 


} 
$db->close(); 


上 述 程序 执行 时 ， 它 会 在 test.db 中 创建 COMPANY 表 ， 并 显示 下 面 所 示 的 消息 : 


Opened database successfully 
Table created successfully 


INSERT 操作 
下 面 的 PHP 程序 显示 了 如 何在 上 面 创建 的 COMPANY 表 中 创建 记录 : 


<?php 
class MyDB extends SQLite3 
{ 
function __construct() 


{ 
} 
} 
$db = new MyDB(); 
if(!$db) { 
echo $db->lastErrorMsg(); 


} else { 
echo "Opened database successfully\n"; 
} 


$sql =<<<EOF 
INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (1, 'Paul', 32, 'California', 20000.00 ); 


$this->open('test.db'); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (2, '‘Allen', 25, 'Texas', 15000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (3, 'Teddy', 23, 'Norway', 20000.00 ); 


INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (4, 'Mark', 25, 'Rich-Mond ', 65000.00 ); 


EOF; 
$ret = $db->exec($sql); 
if('!$ret){ 
echo $db->lastErrorMsg(); 
} else { 


echo "Records created successfully\n"; 


} 
$db->close(); 
?> 


上 述 程序 执行 时 ， 它 会 在 COMPANY 表 中 创建 给 定 记录 ， 并 会 显示 以 下 两 行 : 


Opened database successfully 
Records created successfully 


SELECT 操作 


下 面 的 PHP 程序 显示 了 如 何 从 前 面 创建 的 COMPANY 表 中 获取 并 显示 记录 : 


<?php 


class MyDB extends SQLite3 
{ 


function __construct() 


{ 


} 
} 
$db = new MyDB(); 
if(!$db){ 
echo $db->lastErrorMsg(); 
} else { 
echo "Opened database successfully\n"; 
} 


$sql =<<<EOF 
SELECT * from COMPANY; 


$this->open('test.db'); 


EOF; 


?> 


$ret = $db->query($sql); 


while($row = $ret->fetchArray(SQLITE3_ASSOC) ){ 


echo "ID = ". $row['ID'] . "\n"; 

echo "NAME = ". $row['NAME'] ."\n"; 

echo "ADDRESS = ". $row['ADDRESS'] ."\n"; 
echo "SALARY = ".$row['SALARY'] ."\n\n"; 


} 


echo "Operation done successfully\n"; 
$db->close(); 


Opened database successfully 
ID = 1 

NAME = Paul 

ADDRESS = California 

SALARY = 20000 


ID = 2 

NAME = Allen 
ADDRESS = Texas 
SALARY = 15000 


ID = 3 
NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000 


ID = 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000 


Operation done successfully 


UPDATE 操作 


下 面 的 PHP 代码 显示 了 如 何 使 用 UPDATE 语句 来 更 新 任何 记录 ， 然 后 从 
COMPANY 表 中 获取 并 显示 更 新 的 记录 : 


<?php 
class MyDB extends SQLite3 
{ 


function __construct() 


{ 


} 
} 
$db = new MyDB(); 
if(!$db){ 
echo $db->lastErrorMsg(); 
} else { 
echo "Opened database successfully\n"; 
} 


$sql =<<<EOF 
UPDATE COMPANY set SALARY = 25000.00 where ID=1; 


$this->open('test.db'); 


EOF; 
$ret = $db->exec($sql); 
if(!$ret){ 
echo $db->lastErrorMsg(); 
} else { 
echo $db->changes(), " Record updated successfully\n"; 
} 


$sql =<<<EOF 
SELECT * from COMPANY; 
EOF; 
$ret = $db->query($sql); 
while($row = $ret->fetchArray(SQLITE3_ASSOC) ){ 


echo "ID = ". $row['ID'] . "\n"; 

echo "NAME = ". $row['NAME'] ."\n"; 

echo "ADDRESS = ". $row['ADDRESS'] ."\n"; 
echo "SALARY = ".$row['SALARY'] ."\n\n"; 


} 


echo "Operation done successfully\n"; 
$db->close(); 
?> 


上 述 程序 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
1 Record updated successfully 
ID = 1 

NAME = Paul 

ADDRESS = California 

SALARY = 25000 


ID = 2 

NAME = Allen 
ADDRESS = Texas 
SALARY = 15000 


ID = 3 
NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000 


ID = 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000 


Operation done successfully 


DELETE 操作 


下 面 的 PHP 代码 显示 了 如 何 使 用 DELETE 语句 删除 任何 记录 ， 然 后 从 COMPANY 
表 中 获取 并 显示 剩余 的 记录 : 


<?php 
class MyDB extends SQLite3 
{ 


function __construct() 


{ 


} 
} 
$db = new MyDB(); 
if(!$db){ 
echo $db->lastErrorMsg(); 
} else { 
echo "Opened database successfully\n"; 
} 


$sql =<<<EOF 
DELETE from COMPANY where ID=2; 


$this->open('test.db'); 


EOF; 
$ret = $db->exec($sql); 
if(!$ret){ 
echo $db->lastErrorMsg(); 
} else { 
echo $db->changes(), " Record deleted successfully\n"; 
} 


$sql =<<<EOF 
SELECT * from COMPANY; 
EOF; 
$ret = $db->query($sql); 
while($row = $ret->fetchArray(SQLITE3_ASSOC) ){ 


echo "ID = ". $row['ID'] . "\n"; 

echo "NAME = ". $row['NAME'] ."\n"; 

echo "ADDRESS = ". $row['ADDRESS'] ."\n"; 
echo "SALARY = ".$row['SALARY'] ."\n\n"; 


} 


echo "Operation done successfully\n"; 
$db->close(); 
?> 


上 述 程序 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
1 Record deleted successfully 
ID = 1 

NAME = Paul 

ADDRESS = California 

SALARY = 25000 


ID = 3 
NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000 


ID = 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000 


Operation done successfully 


SQLite - Perl 
ZR 
SQLite3 可 使 用 Perl DBI 模块 与 Perl 进行 集成 。Perl DBI 模块 是 Perl 编程 语言 的 


数据 库 访 问 模块 。 它 定义 了 一 组 提供 标准 数据 库 接 口 的 方法 、 变 量 及 规则 。 
下 面 显 示 了 在 Linux/UNIX 机 器 上 安装 DBI 模块 的 简单 步骤 : 


wget http://search.cpan.org/CPAN/authors/id/T/TI/TIMB/DBI-1.625.1 
tar xvfz DBI-1.625.tar.gz 

cd DBI-1.625 

perl Makefile.PL 

make 

make install 





如 果 您 需要 为 DBI 安装 SQLite 驱动 程序 ， 那 么 可 按照 以 下 步骤 进行 安装 : 


$ wget http://search.cpan.org/CPAN/authors/id/M/MS/MSERGEANT/DBD- S( 
$ tar xvfz DBD-SQLite-1.11.tar.gz 

$ cd DBD-SQLite-1.11 

$ perl Makefile.PL 

$ make 

$ make install 





DBI 接口 API 


以 下 是 重要 的 DBI 程序 ， 可 以 满足 您 在 Per 程序 中 使 用 SQLite 数据 库 的 需求 。 如 
果 您 需要 了 解 更 多 细节 ， 请 查看 Perl DBI 官方 文档 。 


API 
DBI- 
>connect($data_source, 
eae nae, \%attr) 


$dbh->do($sql) 


$dbh->prepare($sql) 


$sth->execute() 


$sth->fetchrow_array/() 


$DBI::err 


$DBI::errstr 


$dbh->disconnect() 


连接 数据 库 


描述 


建立 一 个 到 被 请 求 的 $datasource 的 数据 库 连 接 
或 者 session。 如 果 连 接 成 功 ， 则 返回 一 个 数据 库 
处 理 对 象 。 数据 源 形式 如 下 所 

示 : DBI:SQLite:dbname='test.db'。 其 中 ， 
SQLite 是 SQLite 驱动 程序 名 称 ，test.db 是 
SQLite 数据 库 文件 的 名 称 。 如 果 文 件 名 
_filename 赋值 为 “memory:"， 那 么 它 将 会 在 
RAM 中 创建 一 个 内 存 数据 库 ， 这 只 会 在 session 
的 有 效 时 间 内 持续 。 MRE filename 为 实际 
的 设 各 文件 名 称 ， 那 么 它 将 使 用 这 个 参数 值 尝 试 
打开 数据 库 文 件 。 如 果 该 名 称 的 文件 不 存在 ， 那 
么 将 创建 一 个 新 的 命名 为 该 名 称 的 数据 库 文件 。 
您 可 以 保留 第 二 个 和 第 三 个 参数 为 空白 字符 串 ， 
和 详 见 下 面 的 实 
列 讲解 。 


该 例 程 准备 并 执行 一 个 简单 的 SQL 语句 。 返 回 受 
影响 的 行 数 ， 如 果 发 生 错 误 则 返回 undef。 返 回 值 
-1 意味 着 行 数 未 知 ， 或 不 适用 ， 或 不 可 用 。 在 这 
里 ，$dbh 是 由 DBI->connect() 调用 返回 的 义理 。 


该 例 程 为 数据 库 引 警 后 续 执 行 准 备 一 个 语句 ， 并 
返回 一 个 语句 处 理 对 象 。 


该 例 程 执行 任何 执行 预 准 各 的 语句 需要 的 处 理 。 
如 果 发 生 错 误 则 返回 undef。 如 果 成 功 执 行 ， 则 无 
论 受 影响 的 行 数 是 多 少 ， 总 是 返回 true。 在 这 
里 ，$sth 是 由 $dbh->prepare($sql) 调用 返回 的 语 
Ax, 


该 例 程 获取 下 一 行 数 据 ， 并 以 包含 各 字段 值 的 列 
表 形 式 返 回 。 在 该 列表 中 ，Null 字段 将 作为 undef 
值 返回 。 


这 相当 于 $h->err。 其 中 ，$h 是 任何 的 处 理 类 
型 ， 比 如 $dbh, $sth 或 $drh。 该 程序 返回 最 后 
调用 的 驱动 程序 (driver) 方法 的 数据 库 引 擎 错误 
代码 。 


这 相当 于 $h->errstr。 其 中 ，$h 是 任何 的 处 理 类 
型 ， 比 如 $dbh, $sth 或 $drh。 该 程序 返回 最 后 
调用 的 DBI 方法 的 数据 库 引 警 错 误 消 息 。 


该 例 程 关 闭 之 前 调用 DBI->connect() 打开 的 数据 
库 连 接 。 


下 面 的 Perl 代码 显示 了 如 何 连接 到 一 个 现 有 的 数据 库 。 如 果 数 据 库 不 存在 ， 那 么 它 
就 会 被 创建 ， 最 后 将 返回 一 个 数据 库 对 象 。 


#!/usr/bin/perl 


use DBI; 
use strict; 


my $driver = "SQLite"; 

my $database = "test.db"; 

my $dsn = "DBI: $driver :dbname=$database"; 

my $userid = ""; 

my $password = ""; 

my $dbh = DBI->connect($dsn, $userid, $password, { RaiseError => 1 
or die $DBI::errstr; 


print "Opened database successfully\n"; 





现在 ， 让 我 们 来 运行 上 面 的 程序 ， 在 当前 目录 中 创建 我 们 的 数据 库 test.db。 您 可 
以 根据 需要 改变 路 径 。 保 存 上 面 代 码 到 sqlite.pl 文件 中 ， 并 按 如 下 所 示 执 行 。 如 果 
数据 库 成 功 创建 ， 那 么 会 显示 下 面 所 示 的 消息 : 


$ chmod +x sqlite.pl 
$ ./sqlite.pl 
Open database successfully 


创建 表 
下 面 的 Perl 代码 段 将 用 于 在 先前 创建 的 数据 库 中 创建 一 个 表 : 


#!/usr/bin/perl 


use DBI; 
use strict; 


my $driver "SQLite"; 

my $database "test.db"; 

my $dsn = "DBI: $driver:dbname=$database"; 

my $userid = ""; 

my $password = "": 

my $dbh = DBI- >connect ($dsn, $userid, $password, { RaiseError => 1 
or die $DBI::errstr; 

print "Opened database successfully\n"; 


my $stmt = qq(CREATE TABLE COMPANY 


(ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 
SALARY REAL);); 


my $rv = $dbh->do($stmt); 
if($rv < 0){ 
print $DBI::errstr; 
} else { 
print "Table created successfully\n"; 


$dbh->disconnect(); 
Ne 
上 述 程 序 执行 时 ， 它 会 在 test.db 中 创建 COMPANY 表 ， 并 显示 下 面 所 示 的 消息 : 





Opened database successfully 
Table created successfully 


注意 : 如 果 您 在 任何 操作 中 过 到 了 下 面 的 错误 : in case you see following error in 
any of the operation: 


DBD: :SQLite::st execute failed: not an error(21) at dbdimp.c line : 
BE) 


在 这 种 情况 下 ， 您 已 经 在 DBD-SQLite 安装 中 打开 了 可 用 的 dbdimp.c 文件 ， 找 到 
sqlite3_prepare() EX AX, 并 把 它 的 第 三 个 参数 0 改 为 -1。 最 后 使 用 make 和 
make install 安装 DBD::SQLite， 即 可 解决 问题 。 in this case you will have open 
dbdimp.c file available in DBD-SQLite installation and find out sqlite3_prepare() 
function and change its third argument to -1 instead of 0. Finally install 
DBD::SQLite using make and do make install to resolve the problem. 





INSERT 操作 
FAH Perl 程序 显示 了 如 何在 上 面 创建 的 COMPANY 表 中 创建 记录 : 


#!/usr/bin/perl 


use DBI; 
use strict; 


my $driver = "SQLite"; 

my $database = "test.db"; 

my $dsn = "DBI: $driver :dbname=$database"; 

my $userid = ""; 

my $password = ""; 

my $dbh = DBI->connect($dsn, $userid, $password, { RaiseError => 1 
or die $DBI::errstr; 

print "Opened database successfully\n"; 


my $stmt = qq( INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (1, 'Paul', 32, 'California', 20000.00 )); 
my $rv = $dbh->do($stmt) or die $DBI::errstr; 


$stmt = qq( INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (2, 'Allen', 25, 'Texas', 15000.00 )); 
$rv = $dbh->do($stmt) or die $DBI::errstr; 


$stmt = qq(INSERT INTO COMPANY (ID,NAME, AGE, ADDRESS, SALARY) 
VALUES (3, 'Teddy', 23, 'Norway', 20000.00 )); 
$rv = $dbh->do($stmt) or die $DBI::errstr; 


$stmt = qq(INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) 
VALUES (4, 'Mark', 25, 'Rich-Mond ', 65000.00 );); 
$rv = $dbh->do($stmt) or die $DBI::errstr; 


print "Records created successfully\n"; 
$dbh->disconnect(); 


Sa ogee www ALTE 
上 述 程序 执行 时 ， 它 会 在 COMPANY 表 中 创建 给 定 记录 ， 并 会 显示 以 下 两 行 : 





Opened database successfully 
Records created successfully 


SELECT 操作 


下 面 的 Perl 程序 显示 了 如 何 从 前 面 创 建 的 COMPANY 表 中 获取 并 显示 记录 : 


#!/usr/bin/perl 


use DBI; 
use strict; 


my $driver = "SQLite"; 

my $database = "test.db"; 

my $dsn = "DBI: $driver :dbname=$database"; 

my $userid = ""; 

my $password = ""; 

my $dbh = DBI->connect($dsn, $userid, $password, { RaiseError => 1 
or die $DBI::errstr; 

print "Opened database successfully\n"; 


my $stmt = qq(SELECT id, name, address, salary from COMPANY; ); 
my $sth = $dbh->prepare( $stmt ); 
my $rv = $sth->execute() or die $DBI::errstr; 
if($rv < O){ 
print $DBI::errstr; 


while(my @row = $sth->fetchrow_array()) { 


print "ID = ". $row[0] . "\n"; 

print "NAME = ". $row[1] ."\n"; 
print "ADDRESS = ". $row[2] ."\n"; 
print "SALARY = ". $row[3] ."\n\n"; 


} 
print "Operation done successfully\n"; 
$dbh->disconnect(); 





Opened database successfully 
ID = 1 

NAME = Paul 

ADDRESS = California 

SALARY = 20000 


ID = 2 

NAME = Allen 
ADDRESS = Texas 
SALARY = 15000 


ID = 3 
NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000 


ID = 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000 


Operation done successfully 


UPDATE 操作 


FRA Perl 代码 显示 了 如 何 使 用 UPDATE 语句 来 更 新 任何 记录 ， 然 后 从 
COMPANY 表 中 获取 并 显示 更 新 的 记录 : 


#!/usr/bin/perl 


use DBI; 
use strict; 


my $driver = "SQLite"; 

my $database = "test.db"; 

my $dsn = "DBI: $driver :dbname=$database"; 

my $userid = ""; 

my $password = ""; 

my $dbh = DBI->connect($dsn, $userid, $password, { RaiseError => 1 
or die $DBI::errstr; 

print "Opened database successfully\n"; 


my $stmt = qq(UPDATE COMPANY set SALARY = 25000.00 where ID=1;); 
my $rv = $dbh->do($stmt) or die $DBI::errstr; 
if( $rv < © ){ 
print $DBI::errstr; 
selse{ 
print "Total number of rows updated : $rv\n"; 
} 


$stmt = qq(SELECT id, name, address, salary from COMPANY; ); 
my $sth = $dbh->prepare( $stmt ); 
$rv = $sth->execute() or die $DBI::errstr; 
if($rv < 0){ 
print $DBI::errstr; 


while(my @row = $sth->fetchrow_array()) { 


print "ID = “2 $row[0] = "\n"; 

print "NAME = ". $row[1] ."\n"; 
print "ADDRESS = ". $row[2] ."\n"; 
print "SALARY = ". $row[3] ."\n\n"; 


} 
print "Operation done successfully\n"; 
$dbh->disconnect(); 





Opened database successfully 
Total number of rows updated : 1 
ID = 1 

NAME = Paul 

ADDRESS = California 

SALARY = 25000 


ID = 2 

NAME = Allen 
ADDRESS = Texas 
SALARY = 15000 


ID = 3 
NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000 


ID = 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000 


Operation done successfully 


DELETE 操作 


FHA Perl 代码 显示 了 如 何 使 用 DELETE 语句 删除 任何 记录 ， 然 后 从 COMPANY 
表 中 获取 并 显示 剩余 的 记录 : 


#!/usr/bin/perl 


use DBI; 
use strict; 


my $driver = "SQLite"; 

my $database = "test.db"; 

my $dsn = "DBI: $driver :dbname=$database"; 
my $userid = ""; 

my $password = ""; 


my $dbh = DBI->connect($dsn, $userid, $password, { RaiseError => 1 


or die $DBI::errstr; 
print "Opened database successfully\n"; 


my $stmt = qq(DELETE from COMPANY where ID=2;); 
my $rv = $dbh->do($stmt) or die $DBI::errstr; 
if( $rv = © ){ 

print $DBI::errstr; 
selse{ 

print "Total number of rows deleted : $rv\n"; 
} 


$stmt = qq(SELECT id, name, address, salary from COMPANY; ) ， 


my $sth = $dbh->prepare( $stmt ); 
$rv = $sth->execute() or die $DBI::errstr; 
if($rv < 0){ 

print $DBI::errstr; 


while(my @row = $sth->fetchrow_array()) { 


print "ID = “2 $row[0] = "\n"; 

print "NAME = ". $row[1] ."\n"; 
print "ADDRESS = ". $row[2] ."\n"; 
print "SALARY = ". $row[3] ."\n\n"; 


} 
print "Operation done successfully\n"; 
$dbh->disconnect(); 





Opened database successfully 
Total number of rows deleted 
ID = 1 

NAME = Paul 

ADDRESS = California 

SALARY = 25000 


ID = 3 
NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000 


ID = 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000 


Operation done successfully 


SQLite - Python 
ER 


SQLite3 可 使 用 sqlite3 模块 与 Python 进行 集成 。sqlite3 模块 是 由 Gerhard Haring 
编写 的 。 它 提供 了 一 个 与 PEP 249 描述 的 DB-API 2.0 规范 兼容 的 SQL 接口 。 您 
不 需要 单独 安装 该 模块 ， 因 为 Python 2.5.x 以 上 版 本 默认 自 带 了 该 模块 。 


为 了 使 用 sqlite3 模块 ， 您 首先 必须 创建 一 个 表示 数据 库 的 连接 对 象 ， 然 后 您 可 以 
有 选择 地 创建 光标 对 象 ， 这 将 帮助 您 执行 所 有 的 SQL 语句 。 


Python sqlite3 模块 API 


以 下 是 重要 的 sqlite3 模块 程序 ， 可 以 满足 您 在 Python 程序 中 使 用 SQLite 数据 库 
的 需求 。 如 果 您 需要 了 解 更 多 细节 ， 请 查看 Python sqlite3 模块 的 官方 文档 。 


API 描述 


该 API 打开 一 个 到 SQLite 数 
据 库 文件 database 的 链接 。 
您 可 以 使 用 ":memory:" 来 在 
RAM 中 打开 一 个 到 database 
的 数据 库 连 接 ， 而 不 是 在 磁 瘟 
上 打开 。 如 果 数 据 库 成 功 打 
开 ， 则 返回 一 个 连接 对 象 。 当 
一 个 数据 库 被 多 个 连接 访问 ， 
且 其 中 一 个 修改 了 数据 库 ， 此 
时 SQLite 数据 库 被 锁定 ， 直 
到 事务 提交 。timeout BAR 
示 连 接 等 待 锁 定 的 持续 时 间 ， 
站 到 发 生 异 常 断 开 连 接 。 
timeout 参数 默认 是 5.0 (5 
秒 ) 。 如 果 给 定 的 数据 库 名 称 
filename 不 存在 ， 则 该 调用 将 
创建 一 个 数据 库 。 如 果 您 不 想 
在 当前 目录 中 创建 数据 库 ， 那 
么 您 可 以 指定 带 有 路 径 的 文件 
名 ， 这 样 您 就 能 在 任意 地 方 创 
建 数据 库 。 


该 例 程 创建 一 个 cursor， 将 在 
Python 数据 库 编程 中 用 到 。 该 
方法 接受 一 个 单一 的 可 选 的 参 
数 cursorClass。 如 果 提 供 了 


sqlite3.connect(database [,timeout ,other 
optional arguments]) 


connection.cursor([cursorClass]) 


cursor.execute(sql [, optional 
parameters]) 


connection.execute(sql [, optional 
parameters]) 


cursor.executemany(sql, 


seq_of_parameters) 


connection.executemany(sql[, 
parameters]) 


cursor.executescript(sql_ script) 


connection.executescript(sql_ script) 


connection.total_changes() 


该 参数 ， 则 它 必须 是 一 个 扩展 
自 sqlite3.Cursor 的 自 定义 的 
cursor 类 。 


该 例 程 执行 一 个 SQL 语句 。 
该 SQL 语句 可 以 被 参数 化 

(BI FA ALICE SQL X 
A) o sqlite3 模块 支持 两 种 类 
型 的 占 位 符 : 问好 和 命名 占 位 
符 (命名 样式 ) 。 例如 : 
cursor.execute("insert into 
people values (?, ?)", (who, 
age)) 


该 例 程 是 上 面 执行 的 由 光标 
(cursor) 对 象 提 供 的 方法 的 
快捷 方式 ， 它 通过 调用 光标 
(cursor) 方法 创建 了 一 个 中 
间 的 光标 对 象 ， 然 后 通过 给 定 
的 参数 调用 光标 的 execute 方 


> 二 


该 例 程 对 seq_of_parameters 
中 的 所 有 参数 或 映射 执行 一 个 


SQL MD- 


该 例 程 是 一 个 由 调用 光标 
(cursor) 方法 创建 的 中 间 的 
光标 对 象 的 快捷 方式 ， 然 后 通 
过 给 定 的 参数 调用 光标 的 
executemany 方法 。 


该 例 程 一 旦 接收 到 脚本 ， 会 执 

行 多 个 SQL 语句 。 它 首先 执 

íT COMMIT 语句 ， 然 后 执行 

作为 参数 传 入 的 SQL 脚本 。 

所 有 的 SQL 语句 应 该 用 分 号 
G) 分 隔 。 


该 例 程 是 一 个 由 调用 光标 
(cursor) 方法 创建 的 中 间 的 
光标 对 象 的 快捷 方式 ， 然 后 通 
过 给 定 的 参数 调用 光标 的 
executescript 方法 。 


该 例 程 返回 自 数 据 库 连接 打开 
以 来 被 修改 、 插 入 或 删除 的 数 
据 库 总 行 数 。 

该 方法 提交 当前 的 食物 。 如 果 
您 未 调用 该 方法 ， 那 么 自 您 上 


connection.commit() 一 次 调用 commit() 以 来 所 做 的 
任何 动作 对 其 他 数据 库 连 接 来 
说 是 不 可 见 的 。 
该 方法 回 滚 目 上 一 次 调用 
connection.rollback() commit() 以 来 对 数据 库 所 做 的 
更 改 。 


该 方法 关闭 数据 库 连 接 。 请 注 
意 ， 这 不 会 自动 调用 
commit()。 如 果 您 之 前 未 调用 
connecuome ose commit() 方法 ， 就 直接 关闭 数 
据 库 连接 ， 您 所 做 的 所 有 更 改 
将 全 部 丢失 | 
i 
一 行 ， 返 回 一 个 单一 的 序列 ， 
Wear Mo 当 没有 更 多 可 用 的 数据 时 ， 出 
返回 None。 
该 方法 获取 查询 结果 集中 的 下 
一 行 组 ， 返 回 一 个 列表 。 当 没 


Z S A 47 < l ] 
cursor.fetchmany([size=cursor.arraysize]) Dee Gee 


FX EH size 参数 指定 的 尽 可 能 多 


PA DUR a RS RA 

剩余 ) 的 行 ， 返 回 一 个 列 

Se 志 。 当 没有 可 用 的 行 时 ， 则 返 
回 一 个 空 的 列表 。 


连接 数据 库 


下 面 的 Python 代码 显示 了 如 何 连 接 到 一 个 现 有 的 数据 库 。 如 果 数 据 库 不 存在 ， 那 
么 它 就 会 被 创建 ， 最 后 将 返回 一 个 数据 库 对 象 。 


#!/usr/bin/python 
import sqlite3 
conn = sqlite3.connect('test.db') 
print "Opened database successfully"; 
在 这 里 ， 您 也 可 以 把 数据 库 名 称 复制 为 特定 的 名 称 :memory:， 这 样 就 会 在 RAM 


中 创建 一 个 数据 库 。 现 在 ， 让 我 们 来 运行 上 面 的 程序 ， 在 当前 目录 中 创建 我 们 的 数 
据 库 test.db。 您 可 以 根据 需要 改变 路 径 。 保 存 上 面 代 码 到 sqlite.py 文件 中 ， 并 按 


如 下 所 示 执 行 。 如 果 数 据 库 成 功 创建 ， 那 么 会 显示 下 面 所 示 的 消息 : 


$chmod +x sqlite.py 
$./sqlite.py 
Open database successfully 


创建 表 
下 面 的 Python 代码 段 竺 用 于 在 先前 创建 的 数据 库 中 创建 一 个 表 : 


#!/usr/bin/python 
import sqlite3 


conn = sglite3.connect('test.db') 
print "Opened database successfully"; 


conn.execute('''CREATE TABLE COMPANY 


(ID INT PRIMARY KEY NOT NULL, 
NAME TEXT NOT NULL, 
AGE INT NOT NULL, 
ADDRESS CHAR(50), 

SALARY REAL); ''') 


print "Table created successfully"; 


conn.close() 


上 述 程序 执行 时 ， 它 会 在 test.db 中 创建 COMPANY 表 ， 并 显示 下 面 所 示 的 消息 : 


Opened database successfully 
Table created successfully 


INSERT 操作 


下 面 的 Python 程序 显示 了 如 何在 上 面 创建 的 COMPANY 表 中 创建 记录 : 


#!/usr/bin/python 
import sqlite3 


conn = sqlite3.connect('test.db') 
print "Opened database successfully"; 


conn.execute("INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) \ 
VALUES (1, 'Paul', 32, 'California', 20000.00 )"); 


conn.execute("INSERT INTO COMPANY (ID, NAME, AGE, ADDRESS, SALARY) \ 
VALUES (2, 'Allen', 25, 'Texas', 15000.00 )"); 


conn.execute("INSERT INTO COMPANY (ID,NAME,AGE, ADDRESS, SALARY) \ 
VALUES (3, 'Teddy', 23, 'Norway', 20000.00 )"); 


conn.execute("INSERT INTO COMPANY (ID,NAME,AGE, ADDRESS, SALARY) \ 
VALUES (4, 'Mark', 25, 'Rich-Mond ', 65000.00 )"); 


conn.commit ( ) 
print "Records created successfully"; 
conn.close() 


上 述 程序 执行 时 ， 它 会 在 COMPANY 表 中 创建 给 定 记 录 ， 并 会 显示 以 下 两 行 : 


Opened database successfully 
Records created successfully 


SELECT 操作 


下 面 的 Python 程序 显示 了 如 何 从 前 面 创建 的 COMPANY 表 中 获取 并 显示 记录 : 


#!/usr/bin/python 
import sqlite3 


conn = sgqlite3.connect('test.db') 
print "Opened database successfully"; 


Cursor = conn.execute("SELECT id, name, address, salary from COMP/ 


for row in cursor: 
print "ID = ", row[0] 
print "NAME = ", row[1] 
print "ADDRESS = ", row[2] 
print "SALARY = ", row[3], "\n" 


print "Operation done successfully"; 
conn.close() 





Opened database successfully 
ID= 1 

NAME = Paul 

ADDRESS = California 

SALARY = 20000.0 


ID= 2 

NAME = Allen 
ADDRESS = Texas 
SALARY = 15000.0 


ID = 3 

NAME = Teddy 
ADDRESS = Norway 
SALARY = 20000.0 


ID= 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000.0 


Operation done successfully 


UPDATE 操作 


下 面 的 Python 代码 显示 了 如 何 使 用 UPDATE 语句 来 更 新 任何 记录 ， 然 后 从 


COMPANY 表 中 获取 并 显示 更 新 的 记录 : 


#!/usr/bin/python 
import sqlite3 


conn = sqlite3.connect('test.db') 
print "Opened database successfully"; 


conn.execute("UPDATE COMPANY set SALARY = 25000.00 where ID=1") 
conn.commit 
print "Total number of rows updated :", conn.total_changes 


cursor = conn.execute("SELECT id, name, address, salary from COMP/ 
for row in cursor: 

print "ID = ", row[0] 

print "NAME = ", row[1] 

print "ADDRESS = ", row[2] 

print "SALARY = ", row[3], "\n" 


print "Operation done successfully"; 
conn.close() 


Aoo R 
上 述 程序 执行 时 ， 它 会 产生 以 下 结果 : 


Opened database successfully 
Total number of rows updated : 1 
ID= 1 

NAME = Paul 

ADDRESS = California 

SALARY = 25000.0 


ID= 2 

NAME = Allen 
ADDRESS = Texas 
SALARY = 15000.0 


ID = 3 
NAME = Teddy 

ADDRESS = Norway 
SALARY = 20000.0 


ID= 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000.0 


Operation done successfully 


DELETE 操作 


下 面 的 Python 代码 显示 了 如 何 使 用 DELETE 语句 删除 任何 记录 ， 然 后 从 
COMPANY 表 中 获取 并 显示 剩余 的 记录 : 


#!/usr/bin/python 
import sqlite3 


conn = sqlite3.connect('test.db') 
print "Opened database successfully"; 


conn.execute("DELETE from COMPANY where ID=2;") 
conn.commit 
print "Total number of rows deleted :", conn.total_changes 


cursor = conn.execute("SELECT id, name, address, salary from COMP/ 
for row in cursor: 

print "ID = ", row[0] 

print "NAME = ", row[1] 

print "ADDRESS = ", row[2] 

print "SALARY = ", row[3], "\n" 


print "Operation done successfully"; 
conn.close() 


a] | ah 
上 述 程序 执行 时 ， 它 会 产生 以 下 结果 : 





Opened database successfully 
Total number of rows deleted : 1 
ID= 1 

NAME = Paul 

ADDRESS = California 

SALARY = 20000.0 


ID = 3 
NAME = Teddy 

ADDRESS = Norway 
SALARY = 20000.0 


ID= 4 

NAME = Mark 

ADDRESS = Rich-Mond 
SALARY = 65000.0 


Operation done successfully 


MySQL 教程 


MySQL 教程 


MySQL: 


Mysql 是 最 流行 的 关系 型 数据 库 管理 系统 ， 在 WEB 应 用 方面 MySQL 是 最 好 的 
RDBMS(Relational Database Management System : 关系 数据 库 管理 系统 ) 应 用 软 
件 之 一 。 


在 本 教程 中 ， 会 让 大 家 快速 掌握 Mysql 的 基本 知识 ， 并 轻松 使 用 Mysql| 数 据 库 。 


什么 是 数据 库 ? 


数据 库 (Database) 是 按照 数据 结构 来 组 织 、 存 储 和 管理 数据 的 仓库 ， 


oo ee nee 访问 ， 管 理 ， 搜 索 和 复制 所 保存 
的 数据 。 


我 们 也 可 以 将 数据 存储 在 文件 中 ， 但 是 在 文件 中 读 写 数据 速度 相对 较 慢 。 


所 以 ， 现 在 我 们 使 用 关系 型 数据 库 管理 系统 (RDBMS) 来 存储 和 管理 的 大 数据 
量 。 所 谓 的 关系 型 数据 库 ， 是 建立 在 关系 模型 基础 上 的 数据 库 ， 借 助 于 集合 代数 等 
数学 概念 和 方法 来 处 理 数据 库 中 的 数据 。 


RDBMS 即 关系 数据 库 管 理 系统 (Relational Database Management System) 的 特 
点 : 


o 1. 数 据 以 表格 的 形式 出 现 

e 2. 每 行为 各 种 记录 名 称 

e 3. 每 列 为 记录 名 称 所 对 应 的 数据 域 
e 4. 许 多 的 行 和 列 组 成 一 张 表 单 

e 5. 若 干 的 表单 组 成 database 


RDBMS 术语 


在 我 们 开始 学 习 MySQL 数据 库 前 ， 让 我 们 先 了 解 下 RDBMS 的 一 些 术语 : 
e。 数据 库 : 数据 库 是 一 些 关 联 表 的 集合 。. 


数据 表 : 表 是 数据 的 和 矩阵。 在 一 个 数据 库 中 的 表 看 起 来 像 一 个 简单 的 电子 表 


格 。 
All: 一 列 (数据 元 素 ) 包含 了 相同 的 数据 , 例如 邮政 编码 的 数据 。 
行 : 一 行 (= 元 组 ， 或 记录 ) 是 一 组 相关 的 数据 ， 例 如 一 条 用 户 订阅 的 数据 。 
TUR : 存储 两 倍数 据 ， 宛 余 可 以 使 系统 速度 更 快 。 
ae 主键 是 唯一 的 。 一 个 数据 表 中 只 能 包含 一 个 主键 。 你 可 以 使 用 主键 来 查 
询 数 据 。 
外 键 : 外 键 用 于 关联 两 个 表 。 
合 键 : 复合 键 (组 合 键 ) 将 多 个 列 作为 一 个 索引 键 ， 一 般 用 于 复合 索引 。 


。 索引 : 使 用 索引 可 快速 访问 数据 库 表 中 的 特定 信息 。 索 引 是 对 数据 库 表 中 一 列 


或 多 列 的 值 进行 排序 的 一 种 结构 。 类 似 于 书籍 的 目录 。 
参照 完整 性 : 参照 的 完整 性 要 求 关 系 中 不 允许 引用 不 存在 的 实体 。 与 实体 完整 
性 是 关系 模型 必须 满足 的 完整 性 约束 条 件 ， 目 的 是 保证 数据 的 一 致 性 。 


Mysql 数 据 库 


MySQL 是 一 个 关系 型 数据 库 管 理 系 统 ， 由 瑞典 MySQL AB 公 司 开发 ， 目 前 属于 
Oracle 公 司 。MySQL 是 一 种 关联 数据 库 管理 系统 ， 关联 数据 库 将 数据 保存 在 不 同 的 
表 中 ， 而 不 是 将 所 有 数据 放 在 一 个 大 仓库 内 ， 这 样 就 增加 了 速度 并 提高 了 有 灵活 性 。 


Mysql 是 开源 的 ， 所 以 你 不 需要 支付 额外 的 费用 。 
Mysql 支 持 大 型 的 数据 库 。 可 以 处 理 拥有 上 和 干 万 条 记录 的 大 型 数据 库 。 
MySQL 使 用 标准 的 SQL 数据 语言 形式 。 

Mysql 可 以 允许 于 多 个 系统 上 ， 并 且 支 持 多 种 语言 。 这 些 编程 语言 包括 C、 
C++、Python、Java、Perl、PHP、Eiffel、Ruby 和 Tcl 等 。 
Mysql 对 PHP 有 很 好 的 支持 ，PHP 是 目前 最 流行 的 Web 开 发 语言 。 


e MySQL 支 持 大 型 数据 库 ， 支 持 5000 万 条 记录 的 数据 仓库 ，32 位 系统 表 文 件 最 


大 可 支持 4GB，64 位 系统 支持 最 大 的 表 文件 为 8TB。 
Mysql 是 可 以 定制 的 ， 采 用 了 GPL 协议 ， 你 可 以 修改 源码 来 开发 自己 的 Mysql 系 
Ro 


在 开始 学 习 本 教程 前 你 应 该 了 解 ? 


在 开 


台 学 习 本 教程 前 你 应 该 了 解 PHP 和 HTML 的 基础 知识 ， 并 能 简单 的 应 用 。 


本 教程 的 很 多 例子 都 跟 PHP 语 言 有 关 ， 我 们 的 实例 基本 上 是 采用 PHP 语 言 来 演示 。 
如 果 你 还 不 了 解 PHP， 你 可 以 通过 本 站 的 PHP 教 程 来 了 解 该 语言 。 


MySQL 安装 


所 有 平台 的 Mysql 下 载 地 址 为 : MySQL 下 载 . 挑选 你 需要 的 MySQL Community 
Server 版 本 及 对 应 的 平台 。 


Linux/UNIX 上 安装 Mysql 


Linux 平 台 上 推荐 使 用 RPM 包 来 安装 Mysql,MySQL AB 提 供 了 以 下 RPM 包 的 下 载 地 
址 : 


e MySQL - MySQL 服 务 器 。 你 需要 该 选项 ， 除 非 你 只 想 连 接 运行 在 另 一 台 机 器 
上 的 MySQL 服 务 器 。 

e MySQL-client - MySQL 客户 端 程序 ， 用 于 连接 并 操作 Mysql 服 务 器 。 

e MySQL-devel - 库 和 包含 文件 ， 如 果 你 想 要 编译 其 它 MySQL 客 户 端 ， 例 如 Penl 
模块 ， 则 需要 安装 该 RPM 包 。 

e MySQL-shared - 该 软件 包 包 含 某 些 语言 和 应 用 程序 需要 动态 装载 的 共享 库 
(libmysaqlclient.so*), 4&FAMySQL. 

e MySQL-bench - MySQL 数 据 库 服务 器 的 基准 和 性 能 测试 工具 。 


以 下 安装 Mysql RMP 的 实例 是 在 SuSE Linux 系 统 上 进行 ， 当 然 该 安装 步骤 也 适合 应 
用 于 其 他 支持 RPM 的 Linux 系 统 ， 如 :Centos。 


安装 步骤 如 下 : 

使 用 root 用 户 登 陆 你 的 Linux 系 统 。 

下 载 Mysql RPM 包 ， 下 载 地 址 为 : MySQL 下 载 。 

通过 以 下 命令 执行 Mysql 安 装 ，rpm 包 为 你 下 载 的 rpm 包 : 


[root@host]# rpm -i MySQL-5.0.9-0.1386.rpm 


以 上 安装 mysql 服 务 器 的 过 程 会 创建 mysql 用 户 ， 并 创建 一 个 mysql 配 置 文件 
my.cnf。 


你 可 以 在 /usr/bin 和 /usr/sbin 中 找到 所 有 与 MySQL 相 关 的 二 进 制 文件 。 所 有 数据 表 和 
数据 库 将 在 /varlib/mysql 目 录 中 创建 


以 下 是 一 些 mysql 可 选 包 的 安装 过 程 ， 你 可 以 根据 自己 的 需要 来 安装 


[root@host]# rpm - 
[root@host]# rpm 
[root@host]# rpm 
[root@host]# rpm 


MySQL-client-5.0.9-0.i386.rpm 
MySQL-devel-5.0.9-0.1386.rpm 
MySQL-shared-5.0.9-0.1386.rpm 
MySQL-bench-5.0.9-0.1386.rpm 


I 1 1 
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Window 上 安装 Mysdl 


Window 上 安装 Mysql 相 对 来 说 会 较为 简单 ， 你 只 需要 载 MySQL 下 载 中 下 载 window 
版 本 的 mysqlI 安 装 包 ， 并 解压 安装 包 。 


双击 setup.exe 文件 ， 接 下 来 你 只 需要 安装 默认 的 配置 点 击 "next" 即 可 ， 默 认 情 况 
下 安装 信息 会 在 C:\mysql 目 录 中 。 


接 下 来 你 可 以 通过 "开始 " =》 在 搜索 框 中 输入 " cmd" Hs =) 在 命令 提示 符 上 切换 
到 C:\mysql\bin 目录 ， 并 输入 一 下 命令 : 


mysqld.exe --console 
如 果 安 装 成 功 以 上 命令 将 输出 一 些 mysql 启 动 及 InnoDB 信 息 。 


验证 Mysql 安 装 


在 成 功 安装 Mysql 后 ， 一 些 基础 表 会 表 初 始 化 ， 在 服务 器 启动 后 ， 你 可 以 通过 简单 
的 测试 来 验证 Mysql 是 否 工作 正常 。 


使 用 mysqladmin 工具 来 获取 服务 器 状态 : 


使 用 mysqladmin 命令 俩 检查 服务 器 的 版 本 ,在 linux 上 该 二 进 制 文件 位 于 /usr/bin on 
linux ， 在 window 上 该 二 进 制 文件 位 于 C:\mysql\bin 。 


[root@host]# mysqladmin --version 


linux 上 该 命令 将 输出 以 下 结果 ， 该 结果 基于 你 的 系统 信息 : 

mysqladmin Ver 8.23 Distrib 5.0.9-0, for redhat-linux-gnu on i386 
加 证 一 一 《和 和 
如 果 以 上 命令 执行 后 未 输入 任何 信息 ， 说 明 你 的 Mysql 未 安装 成 功 。 


使 用 MySQL Client(Mysql 客 > in) 执行 简单 的 SQL 命 
AN 


TI 


你 可 以 在 MySQL Client(Mysql 客 > im) 使 用 mysql 命令 连接 到 Mysql 服 务 器 上 ， 软 
认 情 况 下 Mysql 服 务 器 的 密码 为 空 ， 所 以 本 实例 不 需要 输入 密码 。 


命令 如 下 : 


[root@host]# mysql 


以 上 命令 执行 后 会 输出 mysql> 提 示 符 ， 这 说 明 你 已 经 成 功 连接 到 Mysql 服 务 器 上 ， 
你 可 以 在 mysql> 提示 符 执 行 SQL 命令 : 


mysql> SHOW DATABASES; 


Hasso See s5 + 
| Database | 
Poe + 
| mysql | 
| test | 
fos aoe SES + 


2 rows in set (0.13 sec) 


Mysql 23 5 me 2 AY 


Mysql 安 装 成 功 后 ， 默 认 的 root 用 户 密码 为 空 ， 你 可 以 使 用 以 下 命 合 来 创建 root 用 户 
的 密码 : 


[root@host]# mysqladmin -u root password "new_password"; 


现在 你 可 以 通过 以 下 命令 来 连接 到 Mysql 服 务 器 : 


[root@host]# mysql -u root -p 
Enter password: ******* 


注意 : 在 输入 密码 时 ， 密 码 是 不 会 显示 了 ， 你 正确 输入 即 可 。 


Linux Bway Ba MySQL 


如 果 你 需要 在 Linux 系 统 启 启动 时 启动 MySQL 服务 器 ， 你 需要 在 /etc/rc.local 文件 中 
添加 以 下 命 Ap a : 


/etc/init.d/mysqld start 


同样 ， 你 需要 将 mysqld 二 进 制 文件 添加 到 /etc/init.d/ 目录 中 。 


MySQL 管理 


启动 及 关闭 MySQL 服务 器 
首先 ， 我 们 需要 通过 以 下 命令 来 检查 MySQL 服 务 器 是 否 启 动 : 


ps -ef | grep mysqld 


如 果 MySqI 已 经 和 启动， 以 上 命令 将 输出 mysql 进 程 列 表 ， 如 果 mysql 未 和 启动， 你 可 以 
使 用 以 下 命令 来 启动 mysql 服 务 器 : 


root@host# cd /usr/bin 
./safe_mysqld & 


如 果 你 想 关 闭 目前 运行 的 MySQL 服务 器 , 你 可 以 执行 以 下 命 兮 : 


root@host# cd /usr/bin 
./mysqladmin -u root -p shutdown 
Enter password: ****** 


MySQL 用 户 设 置 
如 果 你 需要 添加 MySQL 用 户 ， 你 只 需要 在 mysql 数据 库 中 的 user 表 添 加 新 用 户 
即 可 。 


以 下 为 添加 用 户 的 的 实例 ， 用 户 名 为 guest， 密 码 为 guest123， 并 授权 用 户 可 进行 
SELECT, INSERT 和 UPDATE 操作 权限 : 


root@host# mysql -u root -p 
Enter password: ******* 
mysql> use mysql; 

Database changed 


mysql> INSERT INTO user 
(host, user, password, 
select_priv, insert_priv, update_priv) 
VALUES ('localhost', 'guest', 
PASSWORD('guesti23'), 'Y', 'Y', 'Y'); 
Query OK, 1 row affected (0.20 sec) 


mysql> FLUSH PRIVILEGES; 
Query OK, 1 row affected (0.01 sec) 


mysql> aan host, user, password FROM user WHERE user = 'guest'; 


下 十 
| host 1 user 1 password | 
二 oad opaco acannon paanga a + 
| localhost | guest | er | 

ps Poocqannon e ononon i USO UomoS 十 


1 row in set (0.00 | 
[E 


在 添加 用 户 时 ， 请 注意 使 用 MySQL 提 供 的 PASSWORD() 函数 来 对 密码 进行 加 密 。 
你 可 以 在 以 上 实例 看 到 用 户 密码 加 密 后 为 : 6f8c114b58f2ce9e. 


注意 : 在 注意 需要 执行 FLUSH PRIVILEGES 语句。 这 个 命令 执行 后 会 重新 载 入 授 
权 表 。 如 果 你 不 使 用 该 命令 ， 你 就 无 法 使 用 新 创 建 的 用 户 来 连接 mysql 服 务 器 ， 除 
非 你 重启 mysql 服 务 器 。 


你 可 以 在 创建 用 户 时 ， 为 用 户 指定 权限 ， 在 对 应 的 权限 列 中 ， 在 插入 语句 中 设置 
'Y' 即 可 ， 用 户 权 限 列表 如 下 : 


Select_priv 
Insert_priv 
Update_priv 
Delete_priv 
Create_priv 
Drop_priv 
Reload_priv 
Shutdown_priv 
Process_priv 
File_priv 
Grant_priv 
References_priv 
Index_priv 
Alter_priv 


另外 一 种 添加 用 户 的 方法 为 通过 SQL 的 GRANT 命令 ， 你 下 命令 会 给 指定 数据 订 
TUTORIALS 添 加 用 户 zara ， 密 码 为 zara123 。 


root@host# mysql -u root -p password; 
Enter password: ******* 

mysql> use mysql; 

Database changed 


mysql> GRANT SELECT, INSERT, UPDATE, DELETE, CREATE, DROP 
-> ON TUTORIALS. * 
-> TO 'zara'@'localhost' 
-> IDENTIFIED BY 'zarai23'; 


以 上 命令 会 在 mysql 数 据 库 中 的 user 表 创建 一 条 用 户 信 息 记 录 。 
注意 : MySQL 的 SQL 语句 以 分 号 (;) 作为 结束 标识 。 


/etc/my.cnf 文件 配置 
一 般 情 况 下 ， 你 不 需要 修改 该 配置 文件 ， 该 文件 默认 配置 如 下 : 


[mysqld] 
datadir=/var/lib/mysql 
socket=/var/lib/mysql/mysql.sock 


[mysql.server ] 
user=mysql 
basedir=/var/1lib 


[safe_mysqld] 
err -log=/var/log/mysqld. log 
pid-file=/var/run/mysqld/mysqld.pid 


在 配置 文件 中 ， 你 可 以 指定 不 同 的 错误 日 志文 件 存 放 的 目录 ， 一 般 你 不 需要 改动 这 
些 配置 。 


管理 MySQL 的 命 兮 


以 下 列 出 了 使 用 Mysql 数 据 库 过 程 中 常用 的 命令 : 


e USE 数据 库 名 :选择 要 操作 的 Mysqdl 数 据 库 ， 使 用 该 命令 后 所 有 Mysqdl 命 令 都 只 
针对 该 数据 库 。 

e SHOW DATABASES: 列 出 MySQL 数据 库 管 理 系统 的 数据 库 列 表 。 

e SHOW TABLES: 显示 指定 数据 库 的 所 有 表 ， 使 用 该 命令 前 需要 使 用 use 命 兮 
来 选择 要 操作 的 数据 库 。 

e SHOW COLUMNS FROM 数据 表 : 显示 数据 表 的 属性 ， 属 性 类 型 ， 主 键 信息 


， 是 否 为 NULL， 上 默认 值 等 其 他 信息 。 
e。 SHOW INDEX FROM 数据 表 : 显示 数据 表 的 详细 索引 信息 ， 包 括 PRIMARY 


KEY (主键 ) 。 
e SHOW TABLE STATUS LIKE 数据 表 \G: 该 命令 将 输出 Mysqdl 数 据 库 管理 系统 


的 性 能 及 统计 信息 。 


MySQL PHP 语法 


MySQL 可 应 用 于 多 种 语言 ， 包 括 PERL, C, C++, JAVA 和 PHP. 在 这 些 语言 中 ， 
Mysql 在 PHP 的 web 开 发 中 是 应 用 最 广泛 。 


CERETTA E r, 你 过 你 想 了 解 Mysql 在 PHP 中 的 应 
用 ， 可 以 访问 我 们 的 PHP 中 使 用 Mysdl 介 绍 。 


PHP 提 供 了 多 种 方式 来 访问 和 操作 Mysql 数 据 库 记 录 。PHP Mysql 函 数 格式 如 下 : 


mysql_function(value,value,...); 


以 上 格式 中 function 部 分 描述 了 mysql 范 数 的 功能 ， 如 


mysqli_connect($connect) ; 
mysqli_query($connect, "SQL statement"); 
mysql_fetch_array() 
mysql_connect(),mysql_close() 


以 下 实例 展示 了 PHP 调 用 mysql 函 数 的 语法 : 


<html> 

<head> 

<title>PHP with MySQL</title> 

</head> 

<body> 

<?php 
$retval = mysql_function(value, [value,...]); 
if( !$retval ) 


die ( "Error: a related error message" ); 
} 
// Otherwise MySQL or PHP Statements 
?> 


</body> 
</html> 


从 下 一 章 开始 ， 我 们 将 学 习 到 更 多 的 MySQL 功 能 辑 数 。 


MySQL 连接 
使 用 mysql 二 进 制 方式 连接 


您 可 以 使 用 MySQL 二 进 制 方式 进入 到 mysql 命 令 提 示 符 下 来 连接 MySQL 数 据 库 。 
实例 
以 下 是 从 命令 行 中 连接 mysql 服 务 器 的 简单 实例 : 


[root@host]# mysql -u root -p 

Enter password: ****** 
在 登录 成 功 后 会 出 现 mysql> 命令 提示 窗口 ， 你 可 以 在 上 面 执行 任何 SQL 语句 。 
以 上 命令 执行 后 ， 登 录 成 功 输出 结果 如 下 : 


Welcome to the MySQL monitor. Commands end with ; or \g. 
Your MySQL connection id is 2854760 to server version: 5.0.9 


Type 'help;' or '\h' for help. Type '\c' to clear the buffer. 


在 以 上 实例 中 ， 我 们 使 用 了 root 用 户 登 录 到 mysql 服 务 器 ， 当 然 你 也 可 以 使 用 其 他 
mysql 用 户 登 录 。 


如 果 用 户 权 限 足够 ， 任 何 用 户 都 可 以 在 mysql 的 命令 提示 窗口 中 进行 SQL 操作 。 
退出 mysql> 命令 提示 窗口 可 以 使 用 exit ars, QO RAT : 


mysql> exit 
Bye 


使 用 PHP 脚本 连接 MySQL 


PHP 提供 了 mysql_connect() 函数 来 连接 数据 库 。 
该 轿 数 有 5 个 参数 ， 在 成 功 链接 到 MySQL 后 返回 连接 标识 ， 失 败 返 回 FALSE. 


语法 


connection mysql_connect(server, user, passwd, new_link, client_flag); 





i 一 一 此 
参数 说 明 : 


可 选 。 规 定 要 连接 的 服务 器 。 可 以 包括 端口 号 ， 例 如 
"hostname:port"， 或 者 到 本 地 套 接 字 的 路 径 ， 例 如 对 于 localhost 


AAR B4) ":/path/to/socket", 如 果 PHP 指令 mysal.default_host 未 定义 
(默认 情况 ) ， 则 默认 值 是 localhost:3306'。 
user 可 选 。 用 户 名 。 黑 认 值 是 服务 器 进程 所 有 者 的 用 户 名 。 
passwd 可 选 。 密 码 。 默 认 值 是 空 密码 。 
可 选 。 如 果 用 同样 的 参数 第 二 次 调用 mysql_connect()， 将 不 会 建 
new link ”立新 连接 ， 而 将 返回 已 经 打开 的 连接 标识 。 参 数 new_link 改变 此 
二 行为 并 使 mysql_connect() 总 是 打开 新 的 连接 ， 甚 至 当 
mysql_connect() 便 在 前 面 被 用 同样 的 参数 调用 过 。 
可 选 。client_flags 参数 可 以 是 以 下 常量 的 组 合 : 
MYSQL_CLIENT_SSL - 使 用 SSL 加 密 
client flag MYSQL_CLIENT_COMPRESS - 使 用 压缩 协议 


MYSQL_CLIENT_IGNORE_SPACE - 允许 画 数 名 后 的 间隔 
MYSQL_CLIENT_INTERACTIVE - 允许 关闭 连接 之 前 的 交互 超时 
非 活动 时 间 

你 可 以 使 用 PHP 的 mysql_close() 函数 来 断 开 与 MySQL 数 据 库 的 链接 。 

该 画 数 只 有 一 个 参数 为 mysql_connect() 范 数 创 建 连 接 成 功 后 返回 的 MySQL 连接 标 
识 符 。 

语法 


bool mysql_close ( resource $link_identifier ); 


本 函数 关闭 指定 的 连接 标识 所 关联 的 到 MySQL 服务 器 的 非 持 久 连 接 。 如 果 没 有 指 
定 link_identifier， 则 关闭 上 一 个 打开 的 连接 。 


提示 : 通常 不 需要 使 用 mysql_close()， 因 为 已 打开 的 非 持 久 连 接 会 在 脚本 执行 完 
毕 后 自动 关闭 。 


注释 : mysql_close() 不 会 关闭 由 mysql_pconnect() 建立 的 持久 连接 。 


实例 


你 可 以 尝试 以 下 实例 来 连接 到 你 的 MySQL 服务 器 : 


<html> 

<head> 

<title>Connecting MySQL Server</title> 

</head> 

<body> 

<?php 
$dbhost = 'localhost:3036'; //mysql 服 务 器 主机 地 址 
$dbuser = 'guest'; //mysql 用 户 名 
$dbpass = 'guest123';//mysql 用 户 名 密码 
$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 


echo 'Connected successfully'; 
mysql_close($conn); 

?> 

</body> 

</html> 


MySQL 创建 数据 库 


使 用 mysqladmin 创建 数据 库 


使 用 普通 用 户 ， 你 可 能 需要 特定 的 权限 来 创建 或 者 删除 MySQL 数据 库 。 


所 以 我 们 这 边 使 用 root 用 户 登 录 ，root 用 户 拥有 最 高 权限 ， 可 以 使 用 mysql 
mysqladmin 命令 来 创建 数据 库 。 


实例 
以 下 命令 简单 的 演示 了 创建 数据 库 的 过 程 ， 数 据 名 为 TUTORIALS: 


[root@host]# mysqladmin -u root -p create TUTORIALS 
Enter password: ****** 


以 上 命令 执行 成 功 后 会 创建 MySQL 数据 库 TUTORIALS. 


使 用 PHP 脚 本 创建 数据 库 


PHP 使 用 mysql_query 函数 来 创建 或 者 删除 MySQL 数据 库 。 
该 本 数 有 两 个 参数 ， 在 执行 成 功 时 返回 TRUE， 否则 返回 FALSE. 


语法 


bool mysql query( sql, connection ); 


参数 描述 


sal 必需 。 规 定 要 发 送 的 SQL AW. ER: 查询 字符 串 不 应 以 分 号 
结束 。 


可 选 。 规 定 SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 
的 连接 。 


connection 


实例 
以 下 实例 演示 了 使 用 PHP 来 创建 一 个 数据 库 : 


<html> 


<head> 

<title>Creating MySQL Database</title> 
</head> 

<body> 

<?php 

$dbhost = 'localhost:3036'; 

$dbuser = 'root'; 

$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 
} 
echo 'Connected successfully<br />'; 
$sql = 'CREATE DATABASE TUTORIALS'; 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not create database: ' . mysql_error()); 
echo "Database TUTORIALS created successfully\n"; 
mysql_close($conn); 
?> 
</body> 
</html> 


MySQL 删除 数据 库 


使 用 mysqladmin 删除 数据 库 


ad 普通 用 户 登 陆 mysql 服 务 器 ， 你 可 能 需要 特定 的 权限 来 创建 或 者 删除 MySQL 
JE Eo 


所 以 我 们 这 边 使 用 root 用 户 登 录 ，root 用 户 拥 有 最 高 权限 ， 可 以 使 用 mysql 
mysqladmin 命 命 来 创建 数据 库 。 


在 删除 数据 库 过 程 中 ， 务 必要 十 分 并 愤 ， 因 为 在 执行 删除 命 合 后 ， 所 有 数据 将 会 消 


o 


以 下 实例 删除 数据 库 TUTORIALS( 该 数据 库 在 前 一 章节 已 创建 ) : 


[root@host]# mysqladmin -u root -p drop TUTORIALS 
Enter password: ****** 


执行 以 上 删除 数据 库 命令 后 ， 会 出 现 一 个 提示 框 ， 来 确认 是 否 真 的 删除 数据 库 : 


Dropping the database is potentially a very bad thing to do. 
Any data stored in the database will be destroyed. 


Do you really want to drop the 'TUTORIALS' database [y/N] y 
Database "TUTORIALS" dropped 


(52 FAPHP HAD AS m BRE HEE 


PHP 使 用 mysql_query 函数 来 创建 或 者 删除 MySQL 数据 库 。 
该 回 数 有 两 个 参数 ， 在 执行 成 功 时 返回 TRUE， 否 则 返回 FALSE. 


语法 


bool mysql query( sql, connection ); 


参数 描述 


sal 必需 。 规 定 要 发 送 的 SQL Ai, ER: 查询 字符 串 不 应 以 分 号 
28 Ro 


connection o SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 


实例 


以 下 实例 演示 了 使 用 PHP mysql_query 函 数 来 删除 数据 库 : 


<html> 
<head> 
<title>Deleting MySQL Database</title> 
</head> 
<body> 
<?php 
$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 
$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 
{ 
die(' 连 接 失败 : ' . mysql_error()); 


} 

echo ' 连 接 成 功 <br />'; 

$sql = 'DROP DATABASE TUTORIALS'; 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die(' 删除 数据 库 失 败 : ' . mysql_error()); 


} 

echo "数据 库 TUTORIALS 删除 成 功 \n"，; 
mysql_close($conn); 

?> 

</body> 

</html> 


注意 : 在 使 用 PHP 脚 本 删除 数据 库 时 ， 不 会 出 现 确认 是 否 删 除 信 息 ， 会 直接 删除 指 
定数 据 库 ， 所 以 你 在 删除 数据 库 时 要 特别 小 心 。 


MySQL 选择 数据 库 

在 你 连接 到 MySQL 数据 库 后 ， 可 能 有 多 个 可 以 操作 的 数据 库 ， 所 以 你 需要 选择 你 
要 操作 的 数据 库 。 

从 命令 提 示 窗 口中 选择 MySQL 数 据 库 


在 mysql> 提示 窗口 中 可 以 很 简单 的 选择 特定 的 数据 库 。 你 可 以 使 用 SQL 命令 来 选 
择 指 定 的 数据 库 。 

实例 

以 下 实例 选取 了 数据 库 TUTORIALS: 


[root@host]# mysql -u root -p 
Enter password: ****** 

mysql> use TUTORIALS; 
Database changed 

mysql> 


执行 以 上 命令 后 ， 你 就 已 经 成 功 选择 了 TUTORIALS 数据 库 ， 在 后 续 的 操作 中 都 会 
TE TUTORIALS 数据 库 中 执行 。 


注意 :所 有 的 数据 库 名 ， 表 名 ， 表 字段 都 是 区 分 大 小 写 的 。 所 以 你 在 使 用 SQL 命令 时 
需要 输入 正确 的 名 称 。 


使 用 PHP 脚 本 选择 MySQL 数 据 库 


PHP 提供 了 函数 mysql_select_db 来 选取 一 个 数据 库 。 画 数 在 执行 成 功 后 返回 
TRUE, Gaile] FALSE. 


语法 


bool mysql_select_db( db_name, connection ); 


参数 描述 
db_name 必需 。 规 定 要 选择 的 数据 库 。 
connection ”可 选 。 规 定 MySQL 连接 。 如 果 未 指定 ， 则 使 用 上 一 个 连接 。 


实例 
以 下 实例 展示 了 如 何 使 用 mysql_select db 函数 来 选取 一 个 数据 库 : 


<html> 

<head> 

<title>Selecting MySQL Database</title> 
</head> 
<body> 
<?php 
$dbhost ‘localhost :3036'; 

$dbuser 'guest'; 

$dbpass 'guest123'; 

$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 
} 
echo 'Connected successfully'; 
mysql_select_db( 'TUTORIALS' ); 
mysql_close($conn); 
?> 
</body> 
</html> 


MySQL 数据 类 型 


MySQL 中 定义 数据 字段 的 类 型 对 你 数据 库 的 优化 是 非常 重要 的 。 
MySQL 支 持 多 种 类 型 ， 大 致 可 以 分 为 三 类 : 数值 、 日 期 /时 间 和 字符 串 (字符 ) 类 型 


数值 类 型 
MySQL 支 持 所 有 标准 SQL 数 值 数据 关 型 。 


这 些 类 型 包括 严格 数值 数据 类 型 (INTEGER、SMALLINT、DECIMAL 和 
NUMERIC), 以 及 近似 数值 数据 类 型 (FLOAT、REAL 和 DOUBLE PRECISION)。 


关键 字 INT 是 INTEGER 的 同义词 ， 关 键 字 DEC 是 DECIMAL 的 同义词 。 
BIT 数据 类 型 保存 位 字段 值 ， 并 且 支 持 MyISAM、MEMORY、InnoDB 和 BDB 表 。 


作为 SQL 标准 的 扩展 ，MySQL 也 支持 整数 类 型 TINYINT、MEDIUMINT 和 BIGINT。 
下 面 的 表 显 示 了 需要 的 每 个 整数 类 型 的 存储 和 范围 。 


类 型 大 小 范围 (有 符号 ) ll a] 
符号 途 

小 

RX 

TINYINT 1 字 节 (-128, 127) (0, 255) | 3 
值 

x 

SMALLINT 232% (-32 768, 32767) Ws Oe = 
535) 数 

值 

大 

dey (-8 388 608, 8 388 (0, 16 整 

MEE | See 607) 777215) Æ 
值 

x 

INTs poe (-2 147 483 648, 2 (0, 4294 & 
INTEGER 由 147 483 647) 967 295)  # 
值 

极 

(-9 233 372 036 854 ve ly E 

BIGINT 8 字 节 775 808，9 223 372 mage. F 
036 854 775 807) 551 615) 数 


单 
(-3.402 823 466 o (1.175 | #8 
E+38, 1.175 494 351 494 351 度 


FLOAT 4 字 节 E-38)，0，(1.175 494 E-38， 学 
351 E-38，3.402 823 3.402 823 点 
466 351 E+38) 466 E+38) 数 
值 
(1.797 693 134 862 paras 双 
315 7 E+308，2.225 507 201 4 精 
073 858 507 201 4 E- E-308 度 
DOUBLE 8 Fa 308)，0，(2.225 073 1.797 693 浮 
858 507 201 4 E-308, 134 862 点 
1.797 693 134 862 315 3157 数 
7 E+308) E+308) 值 
对 

DECIMAL(M,D) 小 
DECIMAL ， 如 果 M>D， 依赖 于 M 和 D 的 值 人 数 
为 M+2 人 否则 为 值 

D+2 


日 期 和 时 间 类 型 
表示 时 间 值 的 日 期 和 时 间 类 型 为 DATETIME、DATE、TIMESTAMP、TIME 和 
YEAR。 


每 个 时 间 类 型 有 一 个 有 效 值 范 围 和 一 个 " 需 " 值 ， 当 指定 不 合法 的 MySQL 不 能 表示 的 
值 时 使 用 " 需 " 值 。 


TIMESTAMP 类 型 有 专 有 的 自动 更 新 特性 ， 将 在 后 面 描述 。 


类 型 ( 字 范围 格式 用 途 
节 ) 
Sees g 1000-01-01/9999-12- YYYY-MM- 日 期 各 
31 DD 
TIME 3 '-838:59:597'838:59:59 HH:MM:SS FARR 
续 时 间 
YEAR 1 1901/2155 YYYY 年 份 什 
1000-01-01 YYYY-MM- aa 
DATETIME 8 00:00:00/9999-12-31 DD ae a 和 
23:59:59 HH:MM:SS 
混合 日 期 和 
1970-01-01 YYYYMMDD “YA 
TIMESTAMP 8 00:00:00/2037 年 菜 时 = HHMMSS eT 时 


字符 品类 型 


字符 串 类 型 指 CHAR、VARCHAR、BINARY、VARBINARY、BLOB、TEXT、 
ENUM 和 SET。 该 节 描 述 了 这 些 类 型 如 何 工 作 以 及 如 何在 查询 中 使 用 这 些 类 型 。 


类 型 大 小 用 途 
CHAR 0-255 字 节 EKA 
VARCHAR 0-255 7 KFIR 
TINYBLOB 0-255 7 不 超过 255 个 字符 的 二 进 制 字符 串 
TINYTEXT 0-255 字 节 短文 本 字符 串 
BLOB 0-65 5354 4 二 进 制 形式 的 长 文本 数据 
TEXT 0-65 535 字 节 长 文本 数据 
MEDIUMBLOB ”0-16 777 215 字 节 二 进 制 形式 的 中 等 长 度 文本 数据 
MEDIUMTEXT ”0-16 777 215 字 节 中 等 长 度 文本 数据 
LOGNGBLOB 0-4 294 967 295 字 节 二进制 形式 的 极 大 文本 数据 
LONGTEXT 0-4 294 967 2957F} 极 大 文本 数据 


CHAR 和 VARCHAR 类 型 类 似 ， 但 它们 保存 和 检索 的 方式 不 同 。 它 们 的 最 大 长 度 和 
是 否 尾部 空格 被 保留 等 方面 也 不 同 。 在 存储 或 检索 过 程 中 不 进行 大 小 写 转 换 。 


BINARY 和 VARBINARY 类 类 似 于 CHAR 和 VARCHAR， 不 同 的 是 它们 包含 二 进 制 字 
符 串 而 不 要 非 二 进 制 字 符 串 。 也 就 是 说 ， 它 们 包含 字 节 字符 串 而 不 是 字符 字符 串 。 
这 说 明 它 们 没有 字符 集 ， 并 且 排 序 和 比较 基于 列 值 字 节 的 数值 值 。 


BLOB 是 一 个 二 进 制 大 对 象 ， 可 以 容纳 可 变数 量 的 数据 。 有 4 种 BLOB 类 型 : 
TINYBLOB、BLOB、MEDIUMBLOB 和 LONGBLOB。 它 们 只 是 可 容纳 值 的 最 大 长 
度 不 同 。 


有 4 种 TEXT 类 型 : TINYTEXT、TEXT、MEDIUMTEXT 和 LONGTEXT。 这 些 对 应 4 
种 BLOB 类 型 ， 有 相同 的 最 大 长 度 和 存储 需求 。 


MySQL 创建 数据 表 


创建 MySQL 数 据 表 需要 以 下 信息 : 
© RA 
。 RFRA 
。 定义 每 个 表 字 段 


语法 
以 下 为 创建 MySQL 数 据 表 的 SQL 通用 语法 : 


CREATE TABLE table name (column_name column_type); 


以 下 例子 中 我 们 将 在 TUTORIALS 数据 库 中 创建 数据 表 tutorials_tbl : 


tutorials_tbl( 
tutorial_id INT NOT NULL AUTO_INCREMENT, 
tutorial_title VARCHAR(100) NOT NULL, 
tutorial_author VARCHAR(40) NOT NULL, 
submission_date DATE, 
PRIMARY KEY ( tutorial_id ) 


Ms 
实例 解析 : 
e 如 果 你 不 想 字 段 为 NULL 可 以 设置 字段 的 属性 为 NOT NULL, 在 操作 数据 库 
时 如 果 输 入 该 字段 的 数据 为 NULL ， 就 会 报错 。 
e AUTO_INCREMENT 定 义 列 为 自 增 的 属性 ， 一 般 用 于 主键 ， 数值 会 自动 加 1。 


e PRIMARY KEY 关 键 字 用 于 定义 列 为 主键 。 您 可 以 使 用 多 列 来 定义 主键 ， 列 间 
以 逗号 分 隔 。 


过 命 合 提 示 符 创建 表 


通过 mysql> 命令 窗口 可 以 很 简单 的 创建 MySQL 数 据 表 。 你 可 以 使 用 SQL 语句 
CREATE TABLE 来 创建 数据 表 。 


实例 
以 下 为 创建 数据 表 tutorials_tbl 实例 : 


root@host# mysql -u root -p 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> CREATE TABLE tutorials_tbl( 
-> tutorial_id INT NOT NULL AUTO_INCREMENT, 
-> tutorial_title VARCHAR(100) NOT NULL, 
-> tutorial_author VARCHAR(40) NOT NULL, 
-> submission_date DATE, 
-> PRIMARY KEY ( tutorial_id ) 
m iy 

Query OK, © rows affected (0.16 sec) 

mysql> 


注意 : MySQL 命 令 终 止 符 为 分 号 (;) 。 


使 用 PHP 脚 本 创建 数据 表 


你 可 以 使 用 PHP 的 mysql_query() 本 数 来 创建 已 存在 数据 库 的 数据 表 。 
该 图 数 有 两 个 参数 ， 在 执行 成 功 时 返回 TRUE， 否 则 返回 FALSE. 


语法 


bool mysql_query( sql, connection ); 


参数 描述 
sql 必需 。 规定 要 发 送 的 SQL 查询 。 注释 : 查询 字符 串 不 应 以 分 号 
结束 。 
connection ， 可 选 。 规 定 SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 
的 连接 。 
实例 


以 下 实例 使 用 了 PHP 脚 本 来 创建 数据 表 : 


<html> 


<head> 

<title>Creating MySQL Tables</title> 
</head> 

<body> 

<?php 

$dbhost = 'localhost:3036'; 

$dbuser = 'root'; 

$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 

} 

echo 'Connected successfully<br />'; 

$sql = "CREATE TABLE tutorials_tbl( ". 
"tutorial_id INT NOT NULL AUTO_INCREMENT, ". 
"tutorial_title VARCHAR(100) NOT NULL, ". 
"tutorial_author VARCHAR(40) NOT NULL, ". 
"submission_date DATE, ". 
"PRIMARY KEY ( tutorial_id )); "; 

mysql_select_db( 'TUTORIALS' ); 

$retval = mysql_query( $sql, $conn ); 

if(! $retval ) 


die('Could not create table: ' . mysql_error()); 


echo "Table created successfully\n"; 
mysql_close($conn); 

?> 

</body> 

</html> 


MySQL 删除 数据 表 


MySQL 中 删除 数据 表 是 非常 容易 操作 的 ， 但 是 你 再 进行 删除 表 操 作 时 要 非常 小 
心 ， 因 为 执行 删除 命令 后 所 有 数据 都 会 消失 。 


语法 
以 下 为 删除 MySQL 数 据 表 的 通用 语法 : 


DROP TABLE table_name ; 


在 命令 提示 窗口 中 删除 数据 表 


在 mysql> 命 令 提 示 窗 口中 删除 数据 表 SQL 话 句 为 DROP TABLE : 


实例 
以 下 实例 删除 了 数据 表 tutorials_tbl: 


root@host# mysql -u root -p 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> DROP TABLE tutorials tbl 
Query OK, © rows affected (0.8 sec) 
mysql> 


使 用 PHP 脚 本 删除 数据 表 


PHP 使 用 mysql_query 画 数 来 删除 MySQL 数据 表 。 
该 酌 数 有 两 个 参数 ， 在 执行 成 功 时 返回 TRUE， 否则 返回 FALSE, 
h3> 语法 


bool mysql query( sql, connection ); 


参数 描述 


sal 必需 。 规 定 要 发 送 的 SQL Ai, ER: 查询 字符 串 不 应 以 分 号 
28 Ro 


connection eae SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 


实例 
以 下 实例 使 用 了 PHP 脚 本 删除 数据 表 tutorials_tbl: 


<html> 

<head> 

<title>Creating MySQL Tables</title> 
</head> 
<body> 
<?php 
$dbhost 
$dbuser 'root'; 

$dbpass 'rootpassword'; 

$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


'localhost:3036'; 


die('Could not connect: ' . mysql_error()); 
} 
echo 'Connected successfully<br />'; 
$sql = "DROP TABLE tutorials_tbl"; 
mysql_select_db( 'TUTORIALS' ); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not delete table: ' . mysql_error()); 


echo "Table deleted successfully\n"; 
mysql_close($conn); 

?> 

</body> 

</html> 


MySQL 插入 数据 


MySQL 表 中 使 用 INSERT INTO SQL 语句 来 插入 数据 。 


feo mysql> 命令 提示 窗口 中 向 数据 表 中 插入 数据 ， 或 者 通过 PHP 脚 本 来 插 
入 数据 


语法 
以 下 为 向 MySQL 数 据 表 插入 数据 通用 的 INSERT INTO SQL 语法 : 


INSERT INTO table_name ( fieldi, field2,...fieldN ) 
VALUES 
( valuei, value2,...valueN ); 


如 果 数 据 是 字符 型 ， 必 须 使 用 单 引号 或 者 双 引 号 ， 如 : "value". 
过 命令 提示 窗口 插入 数据 
以 下 我 们 将 使 用 SQL INSERT INTO 语句 向 MySQL 数据 表 tutorials_tbl 插入 数据 


实例 
以 下 实例 中 我 们 将 想 tutorials_tbl 表 插 入 三 条 数据 : 


root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> INSERT INTO tutorials_tbl 
->(tutorial_title, tutorial_author, submission_date) 
->VALUES 
->("Learn PHP", "John Poul", NOW()); 

Query OK, 1 row affected (0.01 sec) 

mysql> INSERT INTO tutorials_tbl 
->(tutorial_title, tutorial_author, submission_date) 
->VALUES 
->("Learn MySQL", "Abdul S", NOW()); 

Query OK, 1 row affected (0.01 sec) 

mysql> INSERT INTO tutorials_tbl 
->(tutorial_title, tutorial_author, submission_date) 
->VALUES 
->("JAVA Tutorial", "Sanjay", '2007-05-06'); 

Query OK, 1 row affected (0.01 sec) 

mysql> 


注意 : 使 用 箭头 标记 (->) 不 是 SQL 语句 的 一 部 分 ， 它 仅仅 表示 一 个 新 行 ， 如 果 一 条 
SQL 语句 太 长， 我 们 可 以 通过 回 车 键 来 创建 一 个 新 行 来 编写 SQL 语句 ，SQL 话 句 的 
命令 结束 符 为 分 号 (i)o 

在 以 上 实例 中 ， 我 们 并 没有 提供 tutorial id 的 数据 ， 因 为 该 字段 我 们 在 创建 表 的 时 
候 已 经 设置 它 为 AUTO_INCREMENT( 自 动 增加 ) 属性 。 所 以 ， 该 字段 会 自动 递增 
而 不 需要 我 们 去 设置 。 实 例 中 NOWO 是 一 个 MySQL 画 数 ， 该 范 数 返 回 日 期 和 时 
间 。 


使 用 PHP 脚 本 插入 数据 


你 可 以 使 用 PHP 的 mysql_query() 函数 来 执行 SQL INSERT INTO 命 令 来 插入 数 
据 。 
该 画 数 有 两 个 参数 ， 在 执行 成 功 时 返回 TRUE， 否则 返回 FALSE. 


语法 


bool mysql query( sql, connection ); 


sal 必需 。 规 定 要 发 送 的 SQL 查询 。 注 释 : 查询 字符 串 不 应 以 分 号 
结束 。 

connection ‘Wits ALE SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 
的 连接 。 

实例 


以 下 实例 中 程序 接收 用 户 输 入 的 三 个 字段 数据 ， 并 插入 数据 表 中 : 


<html> 
<head> 
<title>Add New Record in MySQL Database</title> 
</head> 
<body> 
<?php 
if(isset($_POST['add'])) 
{ 
$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 
$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 
die('Could not connect: ' . mysql_error()); 
} 


if(! get magic quotes gpc() ) 


$tutorial_title = addslashes ($_POST['tutorial_title']); 
$tutorial_author = addslashes ($_POST['tutorial_author']); 
} 
else 
{ 
$tutorial_title = $_POST['tutorial_title']; 
$tutorial_author = $_POST['tutorial_author']; 
} 


$submission_date = $_POST['submission_date']; 


$sql = "INSERT INTO tutorials_tbl ". 
"(tutorial_title, tutorial_author, submission_date) ". 
"VALUES ". 
"('$tutorial_title', '$tutorial_author', '$submission_date')" 
mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not enter data: ' . mysql_error()); 


} 


echo "Entered data successfully\n"; 
mysql_close($conn); 


else 

{ 

?> 

<form method="post" action="<?php $ _PHP_SELF ?>"> 

<table width="600" border="0" cellspacing="1" cellpadding="2"> 
<tr> 

<td width="250">Tutorial Title</td> 

<td> 

<input name="tutorial_ title" type="text" id="tutorial_title"> 
</td> 

</tr> 

<tr> 

<td width="250">Tutorial Author</td> 

<td> 

<input name="tutorial_author" type="text" id="tutorial_author"> 
</td> 

</tr> 

<tr> 

<td width="250">Submission Date [ yyyy-mm-dd ]|</td> 

<td> 

<input name="Submission_date" type="text" id="submission_date"> 
</td> 

</tr> 

<tr> 

<td width="250"> </td> 

<td> </td> 

</tr> 

<tr> 

<td width="250"> </td> 

<td> 

<input name="add" type="Submit" id="add" value="Add Tutorial"> 
</td> 

</tr> 

</table> 

</form> 

<?php 

} 


?> 
</body> 
</html> 


在 我 们 接收 用 户 提交 的 数据 时 ， 为 了 数据 的 安全 性 我 们 需要 使 用 
get_magic_quotes_gpc() 函数 来 判断 特殊 字符 的 转 义 是 否 已 经 开店。 如果 这 个 选项 
为 off (未 开启 ) ， 返 回 0， 那 么 我 们 就 必须 调用 addslashes 这 个 函数 来 为 字符 串 增 
加 转 义 。 


义 。 


比如 邮箱 格式 验证 ， 电 话 号 码 验证 ， 是 否 为 整 
验证 o 


MySQL 坦 询 数据 


MySQL 数据 库 使 用 SQL SELECT 语句 来 查询 数据 。 


be 过 mysql> 命令 提示 窗口 中 在 数据 库 中 查询 数据 ， 或 者 通过 PHP 脚 本 来 查 
询 数 据 


语法 
以 下 为 在 MySQL 数 据 库 中 查询 数据 通用 的 SELECT 语法 : 


SELECT fieldi, field2,...fieldN table_name1i, table_name2... 
[WHERE Clause] 
[OFFSET M ][LIMIT N] 


查询 语句 中 你 可 以 使 用 一 个 或 者 多 个 表 ， 表 之 间 使 用 速 号 (,) 分 割 ， 并 使 用 

WHERE 语句 来 设 定 查 询 条 件 。 

SELECT 命令 可 以 读 取 一 条 或 者 多 条 记录 。 

你 可 以 使 用 星 号 (*) 来 代替 其 他 字段 ，SELECT 语 句 会 返回 表 的 所 有 字段 数据 

你 可 以 使 用 WHERE 语句 来 包含 任何 条 件 。 

J 过 OFFSET 指 SE SELECT 吞 句 开始 查询 的 数据 偏 移 量 。 默 认 情 况 下 偏 
量 为 0。 

。 你 可 以 使 用 LIMIT 属性 来 设 定 返回 的 记录 数 。 


过 命 爷 提 示 符 获取 数据 


以 下 实例 我 们 将 通过 SQL SELECT 命令 来 获取 MySQL 数据 表 tutorials_tbl 的 数 
据 : 


实例 
以 下 实例 将 返回 数据 表 tutorials_tbl 的 所 有 记录 : 


root@host# mysql -u root -p password; 
Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> SELECT * from tutorials_tbl 


人 E 人 TEAREN 
| tutorial_id | tutorial_title | tutorial_author | submission_date 
下 Sy se aaao SS Paoa noana nannan 
| 1 | Learn PHP | John Poul | 2007-05-21 
| 2 | Learn MySQL | Abdul S | 2007-05-21 
| 3 | JAVA Tutorial | Sanjay | 2007-05-21 
Pos oss Sse sao ee PPS a aes boas ssa senosood 


3 rows in set (0.01 sec) 


mysql> 





使 用 PHP 脚 本 来 获取 效 据 


( FAPHPEX XA mysql_query()&SQL SELECT 命令 来 获取 数据 。 


该 轿 数 用 于 执行 SQL 命 爷 ， 然 后 通过 PHP 函数 mysql_fetch_array() 来 使 用 或 输出 
所 有 查询 的 数据 。 


mysql_fetch_array() 画 数 从 结果 集中 取得 一 行 作为 关联 数组 ， 或 数字 数组 ， 或 二 者 
RA 返回 根据 从 结果 集 取 得 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false. 


以 下 实例 为 从 数据 表 tutorials_tbl 中 读 取 所 有 记录 。 


实例 


尝试 以 下 实例 来 显示 数据 表 tutorials_tbl 的 所 有 记录 。 


<?php 
$dbhost 
$dbuser Ootu 

$dbpass 'rootpassword'; 

$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


'localhost:3036'; 


die('Could not connect: ' . mysql_error()); 
$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl'; 
mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 
die('Could not get data: ' . mysql_error()); 
} 
while($row = mysql_ fetch array($retval, MYSQL_ ASSOC)) 
echo "Tutorial ID :{$row['tutorial_id']} <br> ". 
"Title: {$row['tutorial_title']} <br> ". 


"Author: {$row['tutorial_author']} <br> ". 
"Submission Date : {$row['submission_date']} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


以 上 实例 中 ， 读 取 的 每 行 记录 赋值 给 变量 $row， 然 后 再 打印 出 每 个 值 。 
注意 : 记 住 如 果 你 需要 在 字符 串 中 使 用 变量 ， 请 将 变量 置 于 花 括 号 。 


在 上 面 的 例子 中 ，PHP mysql_fetch_array() 函 数 第 二 个 参数 为 MYSQL_ASSOC， 
设置 该 参数 查询 结果 返回 关联 数组 ， 你 可 以 使 用 字段 名 称 来 作为 数组 的 索引 。 


PHP 提 供 了 另外 一 个 函数 mysql_fetch_assoc(), 该 函数 从 结果 集中 取得 一 行 作为 关 
联 数组 。 返回 根据 从 结果 集 取得 的 行 生 成 的 关联 数组 ， 如 果 没 有 更 多 行 ， 则 返回 
false。 


实例 


尝试 以 下 实例 ， 该 实例 使 用 了 mysql_fetch_assoc() 画 数 来 输出 数据 表 tutorial_tbIl 的 
所 有 记录 : 


<?php 


$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 

$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl'; 

mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 

die('Could not get data: ' . mysql_error()); 
} 
while($row = mysql_fetch_assoc($retval)) 

echo "Tutorial ID :{$row['tutorial_id']} <br> ". 
"Title: {$row['tutorial_title']} <br> ". 


"Author: {$row['tutorial_author']} <br> ". 
"Submission Date : {$row['submission_date']} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


你 也 可 以 使 用 常量 MYSQL_NUM 作为 PHP mysql_fetch_array() 函 数 的 第 二 个 参 
数 ， 返 回 数字 数组 。 


实例 


以 下 实例 使 用 MYSQL_NUM 参 数 显示 数据 表 tutorials_tbl 的 所 有 记录 : 


<?php 
$dbhost 
$dbuser Ootu 

$dbpass 'rootpassword'; 

$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


'localhost:3036'; 


die('Could not connect: ' . mysql_error()); 

$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl'; 

mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 

die('Could not get data: ' . mysql_error()); 
} 
while($row = mysql_fetch_array($retval, MYSQL_NUM) ) 

echo "Tutorial ID :{$row[0]} <br>". 
"Title: {$row[1]} <br> ". 


"Author: {$row[2]} <br> ". 
"Submission Date : {$row[3]} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


以 上 三 个 实例 输出 结果 都 一 样 。 


内 存 释 放 


在 我 们 执行 完 SELECT 语 名 后， 释放 游标 内 存 是 一 个 很 好 的 习惯 。 。 可 以 通过 PHP 
函数 mysql_free_result() 来 实现 内 存 的 释放 。 


以 下 实例 演示 了 该 画 数 的 使 用 方法 。 


实例 


党 试 以 下 实例 : 


<?php 


$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 
} 
$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl'; 


mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not get data: ' . mysql_error()); 


} 
while($row = mysql fetch array($retval, MYSQL_NUM)) 
{ 
echo "Tutorial ID :{$row[0]} <br>". 
"Title: {$row[1]} <br> ". 
"Author: {$row[2]} <br> ". 
"Submission Date : {$row[3]} <br> ". 


} 
mysql_free_result($retval); 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


MySQL where 子 句 


我 们 知道 从 MySQL 表 中 使 用 SQL SELECT 语句 来 读 取 数据 。 
如 需 有 条 件 地 从 表 中 选取 数据 ， 可 将 WHERE 子 句 添加 到 SELECT 语句 中 。 


语法 
以 下 是 SQL SELECT 语句 使 用 WHERE 子 句 从 数据 表 中 读 取 数 据 的 通用 语法 : 


SELECT fieldi, field2,...fieldN table_namei1, table_name2... 
[WHERE conditioni [AND [OR]] condition2..... 


查询 语句 中 你 可 以 使 用 一 个 或 者 多 个 表 ， 表 之 间 使 用 到 号 (,) 分 割 ， 并 使 用 
WHERE 语句 来 设 定 查询 条 件 。 

你 可 以 在 WHERE 子 句 中 指定 任何 条 件 。 

你 可 以 使 用 AND 或 者 OR 指定 一 个 或 多 个 条 件 。 
WHERE 子 句 也 可 以 运用 于 SQL 的 DELETE 或 者 UPDATE AS. 

Ae 子 句 类 似 于 程序 语言 中 的 if 条 件 ， 根 据 MySQL 表 中 的 字段 值 来 读 取 指 
定 的 数据 。 


以 下 为 操作 符 列表 ， 可 用 于 WHERE 子 句 中 。 
下 表 中 实例 假定 A 为 10 B 为 20 


学 


作 描 


= 等 号 ， 检 测 两 个 值 是 否 相 等 ， 如 果 相 等 返回 true 


不 等 于 ， 检 测 两 个 值 是 否 相 等 ， 如 果 不 相等 返回 true 
大 于 号 ， 检 测 左边 的 值 是 否 大 于 右边 的 值 , 如 果 左 边 的 值 
大 于 右边 的 值 返回 true 

小 于 号 ， 检 测 左 边 的 值 是 否 小 于 右边 的 值 , 如 果 左 边 的 值 
小 于 右边 的 值 返 回 true 


大 于 等 于 号 ， 检 测 左 边 的 值 是 否 大 于 或 等 于 右边 的 值 , 如 
果 左 边 的 值 大 于 或 等 于 右边 的 值 返 回 true 


o “小 于 等 于 号 ， 检 测 左边 的 值 是 否 小 于 于 或 等 于 右边 的 值 
如 果 左 边 的 值 小 于 或 等 于 右边 的 值 返 回 true 


实例 


(A= B) 返 
回 false。 
(A != B) 返 
回 true. 
(A> B) 返 
回 false。 
(A<B) 返 
回 true。 
(A >= B) 
返回 
false。 

(A <= B) 
返回 
true。 


如 果 我 们 想 再 MySQL 数 据 表 中 读 取 指定 的 数据 ，WHERE 子 句 是 非常 有 用 的 。 


使 用 主键 来 作为 WHERE 子 句 的 条 件 查 询 是 非常 快速 的 。 


如 果 给 定 的 条 件 在 表 中 没有 任何 匹配 的 记录 ， 那 么 查询 不 会 返回 任何 数据 。 


人 命令 提示 符 中 读 取 数据 


我 们 将 在 SQL SELECT 语句 使 用 WHERE 子 句 来 读 取 MySQL 数 据 表 tutorials_tbl 中 


的 数据 : 
实例 


以 下 实例 将 读 取 tutorials_tbl 表 中 tutorial_ author 字段 值 为 Sanjay 的 所 有 记录 : 


root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> SELECT * from tutorials_tbl WHERE tutorial_author='Sanjay'; 


人 E Piao eos sg nanna nnna feo g 5545554055500 
| tutorial_id | tutorial_title | tutorial_author | submission_date 
Poo SS oon ssa Se PSS oo Saga oe aoa Paoa noanoa nanan 
| 3 | JAVA Tutorial | Sanjay | 2007-05-21 

Sao oos sesso sos Moon oss rnea Anon ne EEEE Tonan raaraa aa 


1 rows in set (0.01 sec) 


mysql> 





除非 你 使 用 LIKE 来 比较 字符 串 ， 否 则 MySQL 的 WHERE 子 名 的 字符 串 比较 是 不 区 
分 大 小 写 的 。 你 可 以 使 用 BINARY 关键 字 来 设 定 WHERE 子 名 的 字符 串 比 较 是 区 分 
大 小 写 的 。 
如 下 实例 

root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> SELECT * from tutorials_tbl \ 


WHERE BINARY tutorial_author='sanjay'; 
Empty set (0.02 sec) 


mysql> 


使 用 PHP 脚 本 读 取 数据 


你 可 以 使 用 PHP 画 数 的 mysql_query() 及 相同 的 SQL SELECT # WHERE 子 名 的 
命令 来 获取 数据 。 


该 本 数 用 于 执行 SQL 命令 ， 然 后 通过 PHP EA mysql_fetch_array() 来 输出 所 有 查 
询 的 数据 。 


实例 


以 下 实例 将 从 tutorials_tbl 表 中 返回 使 用 tutorial_author 字段 值 为 Sanjay 的 记录 : 


<?php 


$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 
} 
$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl 
WHERE tutorial_author="Sanjay"'; 


mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not get data: ' . mysql_error()); 


while($row = mysql _fetch_array($retval, MYSQL ASSOC) ) 
{ 
echo "Tutorial ID :{$row['tutorial_id']} <br> ". 
"Title: {$row['tutorial_title']} <br> ". 
"Author: {$row['tutorial_author']} <br> ". 


"Submission Date : {$row['submission_date']} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


MySQL UPDATE 查询 


如 果 我 们 需要 修改 或 更 新 MySQL 中 的 数据 ， 我 们 可 以 使 用 SQL UPDATE 命令 来 操 


Oo :| 


语法 
以 下 是 UPDATE 命令 修改 MySQL 数据 表 数 据 的 通用 SQL 语法 : 


UPDATE table_name SET fieldi=new-valuei, field2=new-value2 
[WHERE Clause] 


e 你 可 以 同时 更 新 一 个 或 多 个 字段 。 
e 你 可 以 在 WHERE 子 句 中 指定 任何 条 件 。 
e 你 可 以 在 一 个 单独 表 中 同时 更 新 数据 。 


当 你 需要 更 新 数据 表 中 指定 行 的 数据 时 WHERE 子 句 是 非常 有 用 的 。 
通过 命令 提示 符 更 新 数据 


以 下 我 们 将 在 SQL UPDATE 命令 使 用 WHERE 子 句 来 更 新 tutorials_tbl 表 中 指定 的 
数据 : 


实例 
以 下 实例 将 更 新 数据 表 中 tutorial id 为 3 的 tutorial_title 字段 值 : 


root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> UPDATE tutorials_tbl 
-> SET tutorial_title='Learning JAVA' 
-> WHERE tutorial_id=3; 

Query OK, 1 row affected (0.04 sec) 

Rows matched: 1 Changed: 1 Warnings: 0 


mysql> 


使 用 PHP 脚 本 更 新 数据 


PHP 中 使 用 函数 mysql_query() 来 执行 SQL 语 句 ， 你 可 以 在 SQL UPDATE 44E 
用 或 者 不 适用 WHERE 子 句 。 


该 图 数 与 在 mysql> 命 合 提 示 符 中 执行 SQL 语 句 的 效果 是 一 样 的 。 


实例 


以 下 实例 将 更 新 tutorial_id 为 3 的 tutorial_title 字段 的 数据 。 


<?php 

$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 

$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 


} 

$sql = 'UPDATE tutorials_tbl 
SET tutorial_title="Learning JAVA" 
WHERE tutorial_id=3'; 


mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not update data: ' . mysql_error()); 
echo "Updated data successfully\n"; 


mysql_close($conn); 
?> 


MySQL DELETE 语句 


你 可 以 使 用 SQL 的 DELETE FROM 命令 来 删除 MySQL 数据 表 中 的 记录 。 
你 可 以 在 mysql> 命 邻 提 示 符 或 PHP 脚 本 中 执行 该 命令 。 


语法 
以 下 是 SQL DELETE 语句 从 MySQL 数 据 表 中 删除 数据 的 通用 语法 : 


DELETE FROM table name [WHERE Clause] 


e 如 果 没 有 指定 WHERE 子 句 ，MySQL 表 中 的 所 有 记录 将 被 删除 。 
e 你 可 以 在 WHERE 子 句 中 指定 任何 条 件 
。 您 可 以 在 单个 表 中 一 次 性 删除 记录 。 


当 你 想 删 除数 据 表 中 指定 的 记录 时 WHERE 子 句 是 非常 有 用 的 。 


人 命令 行 中 删除 数据 


这 里 我 们 将 在 SQL DELETE 命令 中 使 用 WHERE 子 句 来 删除 MySQL 数 据 表 
tutorials_tbl 所 选 的 数据 。 

实例 

以 下 实例 将 删除 tutorial_tbl 表 中 tutorial_id 为 3 的 记录 : 


root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> DELETE FROM tutorials_tbl WHERE tutorial_id=3; 
Query OK, 1 row affected (0.23 sec) 


mysql> 


使 用 PHP 脚本 删除 数据 


PHP 使 用 mysql_query() 函数 来 执行 SQL 语句 ， 你 可 以 在 SQL DELETE 命 令 中 使 用 
或 不 使 用 WHERE 子 句 。 


该 图 数 与 mysql> 命 令 符 执行 SQL 命令 的 效果 是 一 样 的 。 


实例 


以 下 PHP 实 例 将 删除 tutorial_tbl 表 中 tutorial_ id 为 3 的 记录 : 


<?php 

$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 

$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 


} 
$sql = 'DELETE FROM tutorials tbl 


WHERE tutorial_id=3'; 
mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 
die('Could not delete data: ' . mysql_error()); 


echo "Deleted data successfully\n"; 
mysql_close($conn); 
?> 


MySQL LIKE +4] 


我 们 知道 在 MySQL 中 使 用 SQL SELECT 命令 来 读 取 数 据 ， 同时 我 们 可 以 在 
SELECT 语句 中 使 用 WHERE 子 句 来 获取 指定 的 记录 。 


WHERE 子 句 中 可 以 使 用 等 号 (=) 来 设 定 获取 数据 的 条 件 ， 如 "tutorial_author = 
"Sanjay". 

但 是 有 时 候 我 们 需要 获取 tutorial author 字段 含有 "jay" 字符 的 所 有 记录 ， 这 时 我 
们 就 需要 在 WHERE 子 句 中 使 用 SQL LIKE 子 句 。 


SQL LIKE 子 句 中 使 用 百 分 号 (%) 字 符 来 表示 任意 字符 ， 类 似 于 UNIX 或 正则 表达 式 
中 的 星 号 (*)。 


如 果 没 有 使 用 百 分 号 (%), IKE 子 句 与 等 号 (=) 的 效果 是 一 样 的 。 
语法 
以 下 是 SQL SELECT 语句 使 用 LIKE 子 句 从 数据 表 中 读 取 数据 的 通用 语法 : 


SELECT fieldi, field2,...fieldN table namei1i, table_name2... 
WHERE fieldi LIKE condition1 [AND [OR]] filed2 = 'somevalue' 


你 可 以 在 WHERE 子 句 中 指定 任何 条 件 。 

你 可 以 在 WHERE 子 句 中 使 用 LIKE 子 句 。 

你 可 以 使 用 LIKE 子 句 代 替 等 号 (=)。 

LIKE 通常 与 % 一 同 使 用 ， 类 似 于 一 个 元 字符 的 搜索 。 

你 可 以 使 用 AND 或 者 OR 指定 一 个 或 多 个 条 件 。 

你 可 以 在 DELETE 或 UPDATE 命令 中 使 用 WHERE...LIKE 子 句 来 指定 条 件 。 


在 命令 提示 符 中 使 用 LIKE 子 句 


以 下 我 们 将 在 SQL SELECT 命令 中 使 用 WHERE...LIKE 子 句 来 从 MySQL 数 据 表 
tutorials_tbl 中 读 取 数 据 。 


实例 


以 下 是 我 们 将 tutorials_tbl 表 中 获取 tutorial_author 字 段 中 以 "jay" 为 结尾 的 的 所 有 记 
录 : 


root@host# mysql -u root -p password; 
Enter password: ******* 
mysql> use TUTORIALS; 
Database changed 
mysql> SELECT * from tutorials_tbl 
-> WHERE tutorial_author LIKE '%jay'; 
+ 


Pboooposoousads TETEE EEEE EEEE ooo oaaco caooo oaa] 
| tutorial_id | tutorial_title | tutorial_author | submission_date 
Panonan annona 二 二 I i 
| 3 | JAVA Tutorial | Sanjay | 2007-05-21 

TEREE boconsoacooaeceos toa docooeuooeSeouds TERRE 


1 rows in set (0.01 sec) 
mysql> 
图 — S 


在 PHP 脚 本 中 使 用 LIKE $4) 


你 可 以 使 用 PHP 画 数 的 mysql_query() 及 相同 的 SQL SELECT 带 上 WHERE...LIKE 
子 句 的 命令 来 获取 数据 。 


该 本 数 用 于 执行 SQL 命令 ， 然 后 通过 PHP HX mysql_fetch_array() 来 输出 所 有 查 
询 的 数据 。 


但 是 如 果 是 DELETE 或 者 UPDATE 中 使 用 WHERE...LIKE 子 句 的 SQL 语 句 ， 则 无 需 
使 用 mysql_fetch_array() 2X. 





实例 


以 下 是 我 们 使 用 PHP 脚 本 在 tutorials_tbl 表 中 读 取 tutorial_author 字 段 中 以 "jay "为 结 
尾 的 的 所 有 记录 : 


<?php 


$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 
} 
$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl 
WHERE tutorial_author LIKE "%jay%"'; 


mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not get data: ' . mysql_error()); 


while($row = mysql _fetch_array($retval, MYSQL ASSOC) ) 
{ 
echo "Tutorial ID :{$row['tutorial_id']} <br> ". 
"Title: {$row['tutorial_title']} <br> ". 
"Author: {$row['tutorial_author']} <br> ". 


"Submission Date : {$row['submission_date']} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


MySQL 排序 


我 们 知道 从 MySQL 表 中 使 用 SQL SELECT 语句 来 读 取 数据 。 


如 果 我 们 需要 对 读 取 的 数据 进行 排序 ， 我 们 就 可 以 使 用 MySQL 的 ORDER BY 子 句 
来 设 定 你 想 按 哪 个 字段 哪 中 方式 来 进行 排序 ， 再 返回 搜索 结果 。 


语法 
以 下 是 SQL SELECT 语句 使 用 ORDER BY 子 句 特 查询 数据 排序 后 再 返回 数据 : 


SELECT fieldi, field2,...fieldN table_namei, table_name2... 
ORDER BY fieldi, [field2...] [ASC [DESC]] 


你 可 以 使 用 任何 字段 来 作为 排序 的 条 件 ， 从 而 返回 排序 后 的 查询 结果 。 

。 你 可 以 设 定 多 个 字段 来 排序 。 

e 你 可 以 使 用 ASC 或 DESC 关键 字 来 设置 查询 结果 是 按 升序 或 降序 排列 。 默认 
情况 下 ， 它 是 按 升 排列 。 

e 你 可 以 添加 WHERE...LIKE 子 句 来 设置 条 件 。 


在 命令 提示 符 中 使 用 ORDER BY F4 


以 下 将 在 SQL SELECT 语句 中 使 用 ORDER BY 子 句 来 读 取 MySQL 数据 表 
tutorials_tbl 中 的 数据 : 


实例 


党 试 以 下 实例 ， 结 果 将 按 升 序 排列 


root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> SELECT * from tutorials_tbl ORDER BY tutorial_author ASC 


人 E TEOR TEREE 
| tutorial_id | tutorial_title | tutorial_author | submission_date 
ee Se Po SSS 5 soa oe aoa paged 
| 2 | Learn MySQL | Abdul S | 2007-05-24 
| 1 | Learn PHP | John Poul | 2007-05-24 
| 3 | JAVA Tutorial | Sanjay | 2007-05-06 
I oga tange Poe oa sg Sse esses Doonan onoono ae as en nanan 


3 rows in set (0.42 sec) 

mysql> 
TE ea) 
读 取 tutorials_tbl 表 中 所 有 数据 并 按 tutorial_author 字段 的 升序 排列 。 





在 PHP 脚 本 中 使 用 ORDER BY 子 句 


你 可 以 使 用 PHP 画 数 的 mysql_query() 及 相同 的 SQL SELECT # ORDER BY 子 
句 的 命令 来 获取 数据 。 该 本 数 用 于 执行 SQL 命 舍 ， 然 后 通过 PHP WR 
mysql_fetch_array() 来 输出 所 有 查询 的 数据 。 


实例 


尝试 以 下 实例 ， 查 询 后 的 数据 按 tutorial_author 字段 的 降序 排列 后 返回 。 


<?php 


$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 
} 
$sql = 'SELECT tutorial_id, tutorial_title, 
tutorial_author, submission_date 
FROM tutorials_tbl 
ORDER BY tutorial_author DESC'; 


mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not get data: ' . mysql_error()); 


while($row = mysql _fetch_array($retval, MYSQL ASSOC) ) 
{ 
echo "Tutorial ID :{$row['tutorial_id']} <br> ". 
"Title: {$row['tutorial_title']} <br> ". 
"Author: {$row['tutorial_author']} <br> ". 


"Submission Date : {$row['submission_date']} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


Mysql Join 的 使 用 

在 前 几 章 节 中 ， 我 们 已 经 学 会 了 如 果 在 一 张 表 中 读 取 数据 ， 这 是 相对 简单 的 ， 但 是 
在 真正 的 应 用 中 经 常 需要 从 多 个 数据 表 中 读 取 数据 。 

本 章节 我 们 将 向 大 家 介绍 如 何 使 用 MySQL 的 JOIN 在 两 个 或 多 个 表 中 查询 数据 。 
你 可 以 在 SELECT UPDATE 和 DELETE 语句 中 使 用 Mysql 的 join 来 联合 多 表 查 
询 。 

以 下 我 们 将 演示 MySQL LEFT JOIN 和 JOIN 的 使 用 的 不 同 之 处 。 


在 命令 提示 符 中 使 用 JOIN 


我 们 在 TUTORIALS 数 据 库 中 有 两 张 表 tcount tbl 和 tutorials_tbl。 两 张 数据 表 数 据 
如 下 : 


实例 


尝试 以 下 实例 : 


root@host# mysql -u root -p password; 
Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> SELECT * FROM tcount_tbl; 


Poo sos a s5S Sosa 545 peso Sooo soos Sosa ae + 
| tutorial_author | tutorial_count | 

下 十 

| mahran | 20 | 

| mahnaz | NULL | 

| Jen | NULL | 

| Gill | 20 | 

| John Poul | a 

| Sanjay | 1 | 

Dn Danson + 

6 rows in set (0.01 sec) 

mysql> SELECT * from tutorials_tbl; 

全 fos 554556 os Sa555 入 Pondro oaar ap panee y 
| tutorial_id | tutorial_title | tutorial_author | submission_date 
Monodon tanne ee Maona S55 ae eS Se 
| 1 | Learn PHP | John Poul | 2007-05-24 

| 2 | Learn MySQL | Abdul S | 2007-05-24 

| 3 | JAVA Tutorial | Sanjay | 2007-05-06 
全 生生 下 和 Po See eo 下 二 汪汪 人 
3 rows in set (0.00 sec) 

mysql> 


ees 





接 下 来 我 们 就 使 用 MySQL 的 JOIN 来 连接 以 上 两 张 表 来 读 取 tutorials_tbl 表 中 所 有 
tutorial _author 字 段 在 tcount tbl 表 对 应 的 tutorial _count 字 段 值 : 


mysql> SELECT a.tutorial_id, a.tutorial_author, b.tutorial_count 


-> FROM tutorials_tbl a, tcount_tbl b 


-> WHERE a.tutorial_author = b.tutorial_author; 


poses sSooses6 feos Sasa soe seo Ss ee + 
| tutorial_id | tutorial_author | tutorial_count | 
TS 本 SBS non + 
| 1 | John Poul | T 
| 3 | Sanjay | 1 | 
全 Poe oo Sao SSS oe CEFET + 
2 rows in set (0.01 sec) 

mysql> 


在 PHP 脚 本 中 使 用 JOIN 


PHP 中 使 用 mysql _query() 函 数 来 执行 SQL 语句 ， 你 可 以 使 用 以 上 的 相同 的 SQL 语 


HE X mysql_query) HRB 


尝试 如 下 实例 : 


<?php 
$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 
$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 
die('Could not connect: ' . mysql_error()); 


$sql = 'SELECT a.tutorial_id, a.tutorial_author, b.tutorial_count 
FROM tutorials_tbl a, tcount_tbl b 
WHERE a.tutorial_author = b.tutorial_author'; 

mysql_select_db('TUTORIALS'); 

$retval = mysql_query( $sql, $conn ); 

if(! $retval ) 

die('Could not get data: ' . mysql_error()); 
} 
while($row = mysql_fetch_array($retval, MYSQL ASSOC)) 


echo "Author:{$row['tutorial author']} <br> ". 
"Count: {$row['tutorial count']} <br> ". 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


| 
MySQL LEFT JOIN 


MySQL left join 与 join 有 所 不 同 。 MySQL LEFT JOIN 会 读 取 左边 数据 表 的 全 部 数 
据 ， 即 便 右边 表 无 对 应 数据 。 


实例 


尝试 以 下 实例 ， 理 解 MySQL LEFT JOIN 的 应 用 : 


root@host# mysql -u root -p password; 

Enter password: ******* 

mysql> use TUTORIALS; 

Database changed 

mysql> SELECT a.tutorial_id, a.tutorial_author, b.tutorial_count 
-> FROM tutorials_tbl a LEFT JOIN tcount_tbl b 
-> ON a.tutorial_author = b.tutorial_author; 


ee TA A SS Saas ss Bea ee anae + 
| tutorial_id | tutorial_author | tutorial_count | 
Pessoa aos Son SsS soe asa See POSS o ranan nRa + 
| 1 | John Poul | 1 | 
| 2 | Abdul S | NULL | 
| 3 | Sanjay | 1 | 
Se ee ee + 


3 rows in set (0.02 sec) 


以 上 实例 中 使 用 了 LEFT JOIN， 该 语句 会 读 取 左边 的 数据 表 tutorials_tbl 的 所 有 选取 
的 字段 数据 ， 即 便 在 右 侧 表 tcount tbl 中 没有 对 应 的 tutorial_author 字 段 值 。 


MySQL NULL få x 

我 们 已 经 知道 MySQL 使 用 SQL SELECT 命令 及 WHERE 子 句 来 读 取 数据 表 中 的 数 
据 ,但 是 当 提供 的 查询 条 件 字段 为 NULL 时 ， 该 命令 可 能 就 无 法 正常 工作 。 

为 了 义理 这 种 情况 ，MySQL 提 供 了 三 大 运算 符 : 


e IS NULL: 当 列 的 值 是 NULL, 此 运算 符 返 回 true。 
e IS NOT NULL: 当 列 的 值 不 为 NULL, 运算 符 返 回 true。 
e <=>: 比较 操作 符 (不同 于 = 运算 符 ) ， 当 比较 的 的 两 个 值 为 NULL 时 返回 true。 


关于 NULL 的 条 件 比 较 运算 是 比较 特殊 的 。 你 不 能 使 用 = NULL 或 != NULL 在 列 中 
查找 NULL 44 © 


在 MySQL 中 ，NULL 值 与 任何 其 它 值 的 比较 (即使 是 NULL) 永远 返回 false， 即 
NULL = NULL 返回 false 。 


MySQL 中 处理 NULL 使 用 IS NULL 和 IS NOT NULL 运 算 符 。 


在 命令 提示 符 中 使 用 NULL 值 


以 下 实例 中 假设 数据 库 TUTORIALS 中 的 表 tcount tbl 含有 两 列 tutorial_author 和 
tutorial_count, tutorial _count 中 设置 插入 NULL 值 。 


实例 


党 试 以 下 实例 : 


root@host# mysql -u root -p password; 
Enter password: ******* 
mysql> use TUTORIALS; 
Database changed 
mysql> create table tcount_tbl 
-> ( 
-> tutorial_author varchar(40) NOT NULL, 
-> tutorial_count INT 
=); 
Query OK, © rows affected (0.05 sec) 
mysql> INSERT INTO tcount_tbl 
-> (tutorial_author, tutorial_count) values ('mahran', 20); 
mysql> INSERT INTO tcount_tbl 
-> (tutorial_author, tutorial_count) values ('mahnaz', NULL); 
mysql> INSERT INTO tcount_tbl 
-> (tutorial_author, tutorial_count) values ('Jen', NULL); 
mysql> INSERT INTO tcount_tbl 
-> (tutorial_author, tutorial_count) values ('Gill', 20); 


mysql> SELECT * from tcount_tbl; 


Soe ec 0 + 
| tutorial_author | tutorial_count | 
fo6 ses S Jago e sos Fodoranionadi sE + 
| mahran | 20 | 
| mahnaz | NULL | 
| Jen | NULL | 
| Gill | 20 | 
Moppa n oS ose Race Sass sSSoeseescoS5 + 


4 rows in set (0.00 sec) 

mysql> 
Hea 了 al 
以 下 实例 中 你 可 以 看 到 = 和 l= 运算 符 是 不 起 作用 的 : 


mysql> SELECT * FROM tcount tbl WHERE tutorial_count = NULL; 
Empty set (0.00 sec) 
mysql> SELECT * FROM tcount_tbl WHERE tutorial_count != NULL; 
Empty set (0.01 sec) 


查找 数据 表 中 tutorial_count 列 是 否 为 NULL， 必 须 使 用 IS NULL 和 IS NOT NULL, 
如 下 实例 : 


mysql> SELECT * FROM tcount_tbl 
-> WHERE tutorial_count IS NULL; 


ee rora Aa feos aS Soo eee soos + 
| tutorial_author | tutorial_count | 
PSS ae a os Ha Ss ss + 
| mahnaz | NULL | 
| Jen | NULL | 
ee eee ses + 


2 rows in set (0.00 sec) 
mysql> SELECT * from tcount_tbl 
-> WHERE tutorial_count IS NOT NULL; 


Mopo es a Se Ppa See + 
| tutorial_author | tutorial_count | 
fdas ssa Se eS po SS SS ee eee + 
| mahran | 20 | 
| Gill | 20 | 
pee SSS aS Pe SSS + 


2 rows in set (0.00 sec) 


使 用 PHP 上 脚本 处 理 NULL 值 


PHP 脚 本 中 你 可 以 在 if...else 语句 来 处 理 变量 是 否 为 空 ， 并 生成 相应 的 条 件 语句 。 


以 下 实例 中 PHP 设 置 了 $tutorial_count 变 量 ， 然 后 使 用 该 变量 与 数据 表 中 的 
tutorial_count 字段 进行 比较 : 


<?php 


$dbhost = 'localhost:3036'; 
$dbuser = 'root'; 
$dbpass = 'rootpassword'; 


$conn = mysql_connect($dbhost, $dbuser, $dbpass); 
if(! $conn ) 


die('Could not connect: ' . mysql_error()); 


if( isset($tutorial_count )) 


{ 
$sql = 'SELECT tutorial_author, tutorial_count 
FROM tcount_tbl 
WHERE tutorial_count = $tutorial_count'; 
} 
else 
{ 
$sql = 'SELECT tutorial_author, tutorial_count 
FROM tcount_tbl 
WHERE tutorial_count IS $tutorial_count'; 
} 


mysql_select_db('TUTORIALS'); 
$retval = mysql_query( $sql, $conn ); 
if(! $retval ) 


die('Could not get data: ' . mysql_error()); 


while($row = mysql _fetch_array($retval, MYSQL ASSOC) ) 
{ 
echo "Author:{$row['tutorial author']} <br> ". 
"Count: {$row['tutorial count']} <br> ". 


} 


echo "Fetched data successfully\n"; 
mysql_close($conn); 
?> 


MySQL 正则 表达 式 


在 前 面 的 章节 我 们 已 经 了 解 到 MySQL 可 以 通过 LIKE ...% 来 进行 模糊 匹配 。 


MySQL 同样 也 支持 其 他 正则 表达 式 的 匹配 ， MySQL 中 使 用 REGEXP 操作 符 来 进 
行 正 则 表达 式 匹 配 。 


如 果 您 了 解 PHP 或 Perl， 那 么 操作 起 来 就 非常 简单 ， 因 为 MySQL 的 正则 表达 式 匹 配 
与 这 些 脚本 的 类 似 。 


下 表 中 的 正则 模式 可 应 用 于 REGEXP 操作 符 中 。 


模式 


[^] 


p1|p2|p3 


{n} 


{n,m} 


实例 


匹配 输入 字符 串 的 开始 位 置 。 如 果 设 置 了 RegExp 对 象 的 Multiline 
属性 ，^ 也 匹配 \n' 或 \r 之 后 的 位 置 。 


匹配 输入 字符 串 的 结束 位 置 。 如 果 设 置 了 RegExp 对 象 的 Multiline 
属性 ，$ 也 匹配 \n' 或 \F 之 前 的 位 置 。 


匹配 除 "\n" 之 外 的 任何 单个 字符 。 要 匹配 包括 \n' 在 内 的 任何 字 
符 ， 请 使 用 象 [.\n] 的 模式 。 


字符 集合 。 匹 配 所 包含 的 任意 一 个 字符 。 例 如 ， '[abc]' 可 以 匹配 
"plain" 中 的 'a'。 
负 值 字符 集合 。 匹 配 未 包含 的 任意 字符 。 例 如 ， ‘abc! 可 以 匹配 
"plain" 中 的 'p'。 


匹配 p1 或 p2 或 p3。 例 如 ，'zlfood' 能 匹配 "z" 或 
"food". '(z|f)ood' 则 匹配 "zood" 或 "food". 


匹配 前 面 的 子 表达 式 需 次 或 多 次 。 例 如 ，zo 能 匹配 "z" 以 及 
"zoo" 等 价 于 {0,}。 


匹配 前 面 的 子 表达 式 一 次 或 多 次 。 例 如 ，'zo+' 能 匹配 "zo" 以 及 
"zoo"， 但 不 能 匹配 "z" + 等 价 于 {1,}。 


n 是 一 个 非 负 整数 。 匹 配 确定 的 n 次 。 例 如 ，'of2? 不 能 匹配 "Bob" 
中 的 'o'， 但 是 能 匹配 "food" 中 的 两 个 o。 


m 和 mn 均 为 非 负 整数 ， 其 中 n <= m。 最 少 匹 配 n 次 且 最 多 匹配 Mm 


“INO 


了 解 以 上 的 正则 需求 后 ， 我 们 就 可 以 更 加 自己 的 需求 来 编写 带 有 正则 表达 式 的 SQL 
语句 。 以 下 我 们 将 列 出 几 个 小 实例 ( 表 名 : person_tbl ) 来 加 深 我 们 的 理解 : 


查找 name 字 段 中 以 'st 为 开头 的 所 有 数据 : 


mysql> SELECT name FROM person_tbl WHERE name REGEXP 'Ast'; 


查找 name 字 段 中 以 'ok'" 为 结尾 的 所 有 数据 : 


mysql> SELECT name FROM person_tbl WHERE name REGEXP 'ok$'; 


查找 name 字 段 中 包含 ,mar 字 符 串 的 所 有 数据 : 


mysql> SELECT name FROM person_tbl WHERE name REGEXP 'mar'; 


查找 name 字 段 中 以 元 音字 符 开 头 且 以 'ok' 字 符 串 结尾 的 所 有 数据 : 


mysql> SELECT name FROM person_tbl WHERE name REGEXP 'A[aeiou] |ok$ 
E E 





MySQL 事务 


MySQL 事务 主要 用 于 处 理 操 作 量 大 ， 复 杂 度 高 的 数据 。 上 比如 说 ， 在 人 员 管 理 系 统 
中 ， 你 删除 一 个 人 员 ， 你 即 需要 删除 人 员 的 基本 资料 ， 也 要 删除 和 该 人 员 相 关 的 信 
息 ， 如 信箱 ， 文 章 等 等 ， 这 样 ， 这 些 数 据 库 操 作 语 名 就 构成 一 个 事务 ! 


e 在 MySQL 中 只 有 使 用 了 Innodb 数 据 库 引 擎 的 数据 库 或 表 才 支持 事务 

。 事务 处 理 可 以 用 来 维护 数据 库 的 完整 性 ， 保 证 成 批 的 SQL 语句 要 么 全 部 执行 ， 
要 么 全 部 不 执行 

。 事务 用 来 管理 insert,update,delete 语 名 


一 般 来 说 ， 事 务 是 必须 满足 4 个 条 件 (ACID) : Atomicity 〈 原 子 性 ) 、 
Consistency (稳定 性 ) 、lsolation (隔离 性 ) 、Durability (可 靠 性 ) 
。 1、 事 务 的 原子 性 : 一 组 事务 ， 要 么 成 功 ; 要 么 撤回 。 
。2、 稳 定性 : 有 非法 数据 (外 键 约束 之 类 ) ， 事 务 撤回 。 
。 3, RAIE: 事务 独立 运行 。 一 个 事务 处 理 后 的 结果 ， 影 响 了 其 他 事务 ， 那 么 
其 他 事务 会 撤回 。 事 务 的 100% 隔 离 ， 需 要 牺牲 速度 。 
e。4、 可 靠 性 : 软 、 硬 件 崩 溃 后 ，InnoDB 数 据 表 了 驱动 会 利用 日 志文 件 重 构 修改 。 
可 靠 性 和 高 速度 不 可 兼 得 ， innodb flush_log_at_ trx_commit 选 项 决定 什么 时 
候 吧 事务 保存 到 日 志 里 。 


在 Mysql 控 制 台 使 用 事务 来 操作 
1， 开 始 一 个 事务 


start transaction 


2, 做 保存 点 


save point 保存 点 名 称 


3, 操作 
4, WGA, TOURER, RANA, WER, AM MMR. 


PHP 中 使 用 事务 实例 


<?php 

$handler=mysql_connect("localhost","root", "password"); 
mysql_select_db("task"); 

mysql_query("SET AUTOCOMMIT=0");// 设 证 为 不 自动 提交 ， 因 为 MYSQL 默 认 立 即 执 4 
mysql_query ("BEGIN");// 开 始 事 务 定 义 

if(!mysql_query("insert into trans (id) values('2')")) 


和 sse "ROOLBACK" ) ; // 判 断 当 执行 失败 时 回 滚 
if(!mysql_query("insert into trans (id) values('4')")) 
mysql_query ("ROOLBACK" ) ;// 判 断 执 行 失败 回 滚 

人 "COMMIT" ) ; // 执 行事 务 


mysql_close($handler) ; 
?> 
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MySQL ALTERA S 


当 我 们 需要 修改 数据 表 名 或 者 修改 数据 表 字 段 时 ， 就 需要 使 用 到 MySQL ALTERE 


A 
TIo 


开始 本 章 教程 前 让 我 们 先 创建 一 张 表 ， 表 名 为 testalter_ tbl. 


root@host# mysql -u root -p password; 
Enter password: ******* 
mysql> use TUTORIALS; 
Database changed 
mysql> create table testalter_tbl 
-> ( 
-> i INT, 
-> c CHAR(1) 
-> ); 
Query OK, © rows affected (0.05 sec) 
mysql> ee COLUMNS FROM testalter_tbl; 


boo Sao eee es fea aces Hes eS fo ha 55 sso 5 fos a as + 
| Field i Type | Null | Key | Default | Extra | 
eas oS SS SSS fo eo ee re + 
| i | int(11) | YES | | NULL | | 
| € | char(1) | YES | | NULL | | 
下 二 生生 Bae ae ee FE 下 Booo oaoa fess 5 S546 + 


2 rows in set (0.00 sec) 


Hh BR y 添加 或 修 NFS EX 
如 下 命令 使 用 了 ALTER 命令 及 DROP 子 句 来 删除 以 上 创建 表 的 i 字段 : 


mysql> ALTER TABLE testalter_tbl DROP i; 


如 果 数 据 表 中 只 剩余 一 个 字段 则 无 法 使 用 DROP 来 删除 字段 。 


MySQL 中 使 用 ADD 子 句 来 想 数据 表 中 添加 列 ， 如 下 实例 在 表 testalter_tbl 中 添加 
i 字段 ， 并 定义 数据 类 型 : 


mysql> ALTER TABLE testalter_tbl ADD i INT; 


执行 以 上 命令 后 ，i 字段 会 自动 添加 到 数据 表 字 段 的 末尾 


mysql> SHOW COLUMNS FROM testalter_tbl; 


oe pgs oso ooo ee ee 55 + 
| Field | Type | Null | Key | Default | Extra | 
a fea ge ase Pe SSS pS SSS oe + 
Pee | char(1) | YES | | NULL | | 
a as | NULL | | 
了 下 Sea es esha eos oS Ana SES 5555 + 


2 rows in set (0.00 sec) 


如 果 你 需要 指定 新 增 字段 的 位 置 ， 可 以 使 用 MySQL 提 供 的 关键 字 FIRST ( 设 定位 第 
一 列 )， AFTER 字段 名 〈 设 定位 于 某 个 字段 之 后 ) o 


尝试 以 下 ALTER TABLE 语句 , 在 执行 成 功 后 ， 使 用 SHOW COLUMNS 查看 表 结 
构 的 变化 : 


ALTER TABLE testalter_tbl DROP i; 

ALTER TABLE testalter_tbl ADD i INT FIRST; 
ALTER TABLE testalter_tbl DROP i; 

ALTER TABLE testalter_tbl ADD i INT AFTER C; 


FIRST 和 AFTER K# +R SAF ADD 子 句 ， 所 以 如 果 你 想 重 冒 数据 表 字 段 的 位 
置 就 需要 先 使 用 DROP 删除 字段 然后 使 用 ADD 来 添加 字段 并 设置 位 置 。 


修改 字段 类 型 及 名 称 


如 果 需 要 修改 字段 类 型 及 名 称 , 你 可 以 在 ALTER 命 令 中 使 用 MODIFY 或 CHANGE 
子 句 。 


例如 ， 把 字段 c 的 类 型 从 CHAR(1) 改 为 CHAR(10)， 可 以 执行 以 下 命 兮 : 


mysql> ALTER TABLE testalter_tbl MODIFY c CHAR(10); 


使 用 CHANGE £4), 语法 有 很 大 的 不 同 。 在 CHANGE 关键 字 之 后 ， 紧 跟着 的 是 
你 要 修改 的 字段 名 ， 然 后 指定 新 字段 的 类 型 及 名 称 。 尝 试 如 下 实例 : 


mysql> ALTER TABLE testalter_tbl CHANGE i j BIGINT; 


mysql> ALTER TABLE testalter_tbl CHANGE j j INT; 


ALTER TABLE 对 Null 值 和 默认 值 的 影响 


当 你 修改 字段 时 ， 你 可 以 指定 是 否 包含 只 或 者 是 否 设置 默认 值 。 
以 下 实例 ， 指 定 字段 j 为 NOT NULL 且 默 认 值 为 100 。 


mysql> ALTER TABLE testalter_tbl 
-> MODIFY j BIGINT NOT NULL DEFAULT 100; 


如 果 你 不 设置 默认 值 ，MySQL 会 自动 设置 该 字段 默认 为 NULL。 


修改 字段 默认 值 
你 可 以 使 用 ALTER 来 修改 字段 的 默认 值 ， 尝 试 以 下 实例 : 


mysql> ALTER TABLE testalter_tbl ALTER i SET DEFAULT 1000; 
mysql> SHOW COLUMNS FROM testalter_tbl; 
+ + 


Be See ee gt Se ee ee 
| Field | Type | Null | Key | Default | Extra | 
PSs so peo Sa SS SPS ao Maassa 0 (eae 55 + 
ee | char(1) | YES | | NULL | | 
a [anita (elses ss | 1000 | | 
ee Has ss 555 sss ee Aaaa Maana 55 + 


2 rows in set (0.00 sec) 


你 也 可 以 使 用 ALTER 命令 及 DROP 子 名 来 删除 字段 的 默认 值 ， 如 下 实例 : 


mysql> ALTER TABLE testalter_tbl ALTER i DROP DEFAULT; 
mysql> SHOW COLUMNS FROM testalter_tbl; 
+ 


下 pessoas ee I Se SS 55555 Aroan oan + 
| Field | Type | Null | Key | Default | Extra | 
oso fo Soe hoes ee tees + 
| @ | char(1) | YES | | NULL | | 
| i nE CE Mes | NULL | | 
Se es Paseo Hoa aS Ses asscc4 二 三 三 < 十 


2 rows in set (0.00 sec) 
Changing a Table Type: 


修改 数据 表 类 型 ， 可 以 使 用 ALTER 命令 及 TYPE 子 句 来 完成 。 党 试 以 下 实例 ， 我 
们 将 表 testalter tbl 的 类 型 修改 为 MYISAM : 


注意 : 查看 数据 表 类 型 可 以 使 用 SHOW TABLE STATUS 语句 。 


mysql> ALTER TABLE testalter_tbl TYPE = MYISAM; 
mysql> SHOW TABLE STATUS LIKE 'testalter_tbl'\G 
rR REREREREREEREEREREREEkEkK JIN row WR 
Name: testalter_tbl 
Type: MyISAM 
Row_format: Fixed 
Rows: 0 
Avg_row_length: 0 
Data_length: 0 
Max_data_length: 25769803775 
Index_length: 1024 
Data_free: 0 
Auto_increment: NULL 
Create_time: 2007-06-03 08:04:36 
Update_time: 2007-06-03 08:04:36 
Check_time: NULL 
Create_options: 
Comment : 
1 row in set (0.00 sec) 


修改 表 名 


如 果 需 要 修改 数据 表 的 名 称 ， 可 以 在 ALTER TABLE 语句 中 使 用 RENAME FAR 
实现 。 


尝试 以 下 实例 将 数据 表 testalter_tbl 重 命名 为 alter_tbl : 


mysql> ALTER TABLE testalter_tbl RENAME TO alter_tbl; 


ALTER 命令 还 可 以 用 来 创建 及 删除 MySQL 数 据 表 的 索引 ， 该 功能 我 们 会 在 接 下 来 


MySQL 35| 
MySQLSRAI ÉJT F MySQL BM TEREEN, RS TUAAWSMySOL 
的 检索 速度 。 


打 个 比方 ， 如 果 合 理 的 设计 且 使 用 索引 的 MySQL 是 一 辆 兰博基尼 的 话 ， 那 么 没有 设 
计 和 使 用 索引 的 MySQL 就 是 一 个 人 力 三 轮 车 。 


索引 分 单列 索引 和 组 合 索 引 。 单 列 素 引 ， 即 一 个 索引 只 包含 单个 列 ， 一 个 表 可 以 有 
多 个 单列 索引 ， 但 这 不 是 组 合 索 引 。 组 合 索 引 ， 即 一 个 索 包 含 多 个 列 。 


创建 索引 时 ， 你 需要 确保 该 索引 是 应 用 在 SQL 查询 语句 的 条 件 (一 般 作 为 WHERE 
子 句 的 条 件 )。 


实际 上 ， 索 引 也 是 一 张 表 ， 该 表 保存 了 主键 与 索引 字段 ， 并 指向 实体 表 的 记录 。 


上 面 都 在 说 使 用 索引 的 好 勾 ， 但 过 多 的 使 用 索引 将 会 造成 滥用 。 因 此 素 引 也 会 有 它 
的 缺点 : 虽然 素 引 大 大 提高 了 查询 速度 ， 同 时 却 会 降低 更 新 表 的 速度 ， 如 对 表 进 行 
INSERT、UPDATE 和 DELETE。 因 为 更 新 表 时 ，MySQL 不 仅 要 保存 数据 ， 还 要 保 
存 一 下 索引 文件 。 


建立 索引 会 占用 磁盘 空间 的 索引 文件 。 


普通 索引 


创建 索引 
这 是 最 基本 的 索引 ， 它 没有 任何 限制 。 它 有 以 下 几 种 创建 方式 : 


CREATE INDEX indexName ON mytable(username(length) ); 


如 果 是 CHAR，VARCHAR 类 型 ，length 可 以 小 于 字段 实际 长 度 ; 如 果 是 BLOB 和 
TEXT 类 型 ， 必 须 指 定 length。 


ALTER mytable ADD INDEX [indexName] ON (username(length) ) 


O FE FRAY oy ER SH FE 


CREATE TABLE mytable( 

ID INT NOT NULL, 

username VARCHAR(16) NOT NULL, 

INDEX [indexName] (username(length) ) 


); 


删除 索引 的 语 ; 


DROP INDEX [indexName] ON mytable; 


唯一 乘 引 


它 与 前 面 的 普通 索引 类 似 ， 不 同 的 就 是 : 索引 列 的 值 必 须 唯 一 ， 但 允许 有 空 值 。 如 
果 是 组 合 索 引 ， 则 列 值 的 组 合 必须 唯一 。 它 有 以 下 几 种 创建 方式 : 


创建 索引 


CREATE UNIQUE INDEX indexName ON mytable(username(length) ) 


ALTER mytable ADD UNIQUE [indexName] ON (username(length) ) 


创建 表 的 时 候 直 接 指定 


CREATE TABLE mytable( 

ID INT NOT NULL, 

username VARCHAR(16) NOT NULL, 

UNIQUE [indexName] (username(length) ) 


); 


使 用 ALTER 命令 添加 和 删除 索引 


有 四 种 方式 来 添加 数据 表 的 索引 : 

e ALTER TABLE tbl_name ADD PRIMARY KEY (column_list): 该 语句 添加 一 
个 主键 ， 这 意味 着 索引 值 必须 是 唯一 的 ， 且 不 能 为 NULL。 

e ALTER TABLE tbl_name ADD UNIQUE index_name (column_list): 这 条 话 
句 创 建 索 引 的 值 必 须 是 唯一 的 (除了 NULL 外 ，NULL 可 能 会 出 现 多 次 ) 。 

e ALTER TABLE tbl_name ADD INDEX index_name (column_list): 添加 普通 
RSI, RIGZTHMSR, 

e ALTER TABLE tbl_name ADD FULLTEXT index_name (column_list):i% i2 
句 指定 了 索引 为 FULLTEXT, AFERSI 


以 下 实例 为 在 表 中 添加 索引 。 


mysql> ALTER TABLE testalter_tbl ADD INDEX (c); 


你 还 可 以 在 ALTER 命令 中 使 用 DROP 子 句 来 删除 索引 。 党 试 以 下 实例 删除 索引 : 


mysql> ALTER TABLE testalter_tbl DROP INDEX (c); 


使 用 ALTER 命令 添加 和 删除 主键 


主键 只 能 作用 于 一 个 列 上 ， 添 加 主键 索引 时 ， 你 需要 确保 该 主键 默认 不 为 空 (NOT 
NULL) 。 实 例如 下 : 


mysql> ALTER TABLE testalter_tbl MODIFY i INT NOT NULL; 
mysql> ALTER TABLE testalter_tbl ADD PRIMARY KEY (i); 


你 也 可 以 使 用 ALTER 命令 删除 主键 : 


mysql> ALTER TABLE testalter_tbl DROP PRIMARY KEY; 
删除 指定 时 只 需 指 定 PRIMARY KEY， 但 在 删除 索引 时 ， 你 必须 知道 索引 名 。 


显示 埃 引 信息 


你 可 以 使 用 SHOW INDEX 命令 来 列 出 表 中 的 相关 的 索引 信息 。 可 以 通过 添加 \G 
来 格式 化 输出 信息 。 


党 试 以 下 实例 : 


mysql> SHOW INDEX FROM table_name\G 


MySQL 临时 表 


MySQL 临时 表 在 我 们 需要 保存 一 些 临 时 数据 时 是 非常 有 用 的 。 临 时 表 只 在 当前 连 
接 可 见 ， 当 关闭 连接 时 ，Mysql 会 自动 删除 表 并 释放 所 有 空间 。 


临时 表 在 MySQL 3.23 版 本 中 添加 ， 如 果 你 的 MySQL 版 本 低 于 3.23 版 本 就 无 法 使 用 
MySQL 的 临时 表 。 不 过 现在 一 般 很 少 有 再 使 用 这 么 低 版 本 的 MySQL 数 据 库 服务 


o 


MySQL 临 时 表 只 在 当前 连接 可 见 ， 如 果 你 使 用 PHP 脚 本 来 创建 MySQL 临 时 表 ， 那 
没 当 PHP 脚 本 执行 完成 后 ， 该 临时 表 也 会 自动 销毁 。 


如 果 你 使 用 了 其 他 MySQL 客 户 端 程序 连接 MySQL 数 据 库 服务 器 来 创建 临时 表 ， 那 
么 只 有 在 关闭 客户 端 程序 时 才 会 销毁 临时 表 ， 当 然 你 也 可 以 手动 销毁 。 


实例 


以 下 展示 了 使 用 MySQL 临时 表 的 简单 实例 ， 以 下 的 SQL 代码 可 以 适用 于 PHP 脚 本 
的 mysql_query() 画 数 。 


mysql> CREATE TEMPORARY TABLE SalesSummary ( 
-> product_name VARCHAR(50) NOT NULL 
-> , total_sales DECIMAL(12,2) NOT NULL DEFAULT 0.00 


-> , avg_unit_price DECIMAL(7,2) NOT NULL DEFAULT 0.00 
-> total_units_sold INT UNSIGNED NOT NULL DEFAULT 0 


); 
Query OK, © rows affected (0.00 sec) 


mysql> INSERT INTO SalesSummary 
-> (product_name, total_sales, avg_unit_price, total_units_sol¢ 
-> VALUES 
-> ('cucumber', 100.25, 90, 2); 


mysql> SELECT * FROM SalesSummary; 
+ 


Sasa ose oa fea 963 5 S555 ena aoaaa eS ssc ses 
| product_name | total_sales | avg_unit_price | total_units_sold | 
Pass oS 5o5555e 55 和 Ee PaaS oso ses Se + 
| cucumber | 100.25 | 90.00 | 2 | 
Moo og se BPS oS aS Popes SSS oann PP 555 SS ease + 


1 row in set (0.00 sec) 


当 你 使 用 SHOW TABLES 命 邻 显示 数据 表 列 表 时 ， 你 将 无 法 看 到 SalesSummary 


o 





如 果 你 退出 当前 MySQL 会 话 ， 再 使 用 SELECT 命令 来 读 取 原 先 创 建 的 临时 表 数 
据 ， 那 你 会 发 现 数据 库 中 没有 该 表 的 存在 ， 因 为 在 你 退出 时 该 临时 表 已 经 被 销毁 
了 。 


删除 MySQL 临时 表 


默认 情况 下 ， 当 你 断 开 与 数据 库 的 连接 后 ， 临 时 表 就 会 自动 被 销毁 。 当 然 你 也 可 以 
在 当前 MySQL 会 话 使 用 DROP TABLE 命令 来 手动 删除 临时 表 。 


以 下 是 手动 删除 临时 表 的 实例 : 


mysql> CREATE TEMPORARY TABLE SalesSummary ( 
-> product name VARCHAR(50) NOT NULL 
-> , total sales DECIMAL(12,2) NOT NULL DEFAULT 0.00 
-> , avg_unit_price DECIMAL(7,2) NOT NULL DEFAULT 0.00 
-> total_units_sold INT UNSIGNED NOT NULL DEFAULT 0 


); 
Query OK, © rows affected (0.00 sec) 


mysql> INSERT INTO SalesSummary 
-> (product_name, total_sales, avg_unit_price, total_units_solc 
-> VALUES 
-> ('cucumber', 100.25, 90, 2); 


mysql> SELECT * FROM SalesSummary; 


Paopao dangoa aoa Pbocodgecaendscgs bossocnuscongescas osoousecopccocousas + 
| product_name | total_sales | avg_unit_price | total_units_sold | 
站 or a nn Moona anano nnna anaa + 
| cucumber | 100.25 | 90.00 | 2 | 
Poouecasdeodous Oonace TEREFE JPoanodood goood aan + 


1 row in set (0.00 sec) 

mysql> DROP TABLE SalesSummary; 

mysql> SELECT * FROM SalesSummary; 

ERROR 1146: Table 'TUTORIALS.SalesSummary' doesn't exist 


EE 





MySQL 复制 表 


如 果 我 们 需要 完全 的 复制 MySQL 的 数据 表 ， 包 括 表 的 结构 ， 索 引 ， 默 认 值 等 。 如 
果 仅 仅 使 用 CREATE TABLE ... SELECT 命令 ， 是 无 法 实现 的 。 


本 章节 将 为 大 家 介绍 如 何 完整 的 复制 MySQL 数 据 表 ， 步 又 如 下 : 


。 使 用 SHOW CREATE TABLE 命令 获取 创建 数据 表 (CREATE TABLE) 语句 ， 
该 语句 包含 了 原 数据 表 的 结构 ， 索 引 等 。 


。 复制 以 下 命令 显示 的 SQL 语句 ， 修 改 数据 表 名 ， 并 执行 SQL 语句 ， 通 过 以 上 命 
D 将 完全 的 复制 数据 表 结 构 。 


e。 如 果 你 想 复制 表 的 内 容 ， 你 就 可 以 使 用 INSERT INTO... SELECT 语句 来 实 
现 。 


实例 

尝试 以 下 实例 来 复制 表 tutorials_tbl 。 
步骤 一 : 

获取 数据 表 的 完整 结构 。 


mysql> SHOW CREATE TABLE tutorials_tbl \G; 
kkxkxkxkkkkxkxkkkkkkkkkkkkkkkkxkk 1\. row 类 类 类 类 炎炎 炎炎 类 炎 炎炎 炎炎 炎炎 炎炎 炎炎 类 火炎 炎炎 类 类 
Table: tutorials tbl 

Create Table: CREATE TABLE ‘tutorials tbl ( 
tutorial id int(11) NOT NULL auto_increment, 
`tutorial_title` varchar(100) NOT NULL default '', 
`tutorial_author` varchar(40) NOT NULL default '', 
“submission date date default NULL, 
PRIMARY KEY (“tutorial id `)， 
UNIQUE KEY AUTHOR_INDEX (tutorial author ) 

) TYPE=MyISAM 

1 row in set (0.00 sec) 


ERROR: 
No query specified 
步骤 二 : 
修改 SQL 语句 的 数据 表 名 ， 并 执行 SQL 语句 。 


mysql> CREATE TABLE ‘clone tbl ( 
-> tutorial id int(11) NOT NULL auto_increment, 
-> `tutorial_title` varchar(100) NOT NULL default '', 
-> tutorial author varchar(40) NOT NULL default '', 
-> ~submission_date’ date default NULL, 
-> PRIMARY KEY ( tutorial id )， 
-> UNIQUE KEY “AUTHOR_INDEX ”( tutorial author ) 

-> ) TYPE=MyISAM; 

Query OK, © rows affected (1.80 sec) 


步骤 三 : 


执行 完 第 二 步骤 后 ， 你 将 在 数据 库 中 创建 新 的 克隆 表 clone tbl, 如 果 你 想 拷 贝 数 
据 表 的 数据 你 可 以 使 用 INSERT INTO... SELECT 语句 来 实现 。 


mysql> INSERT INTO clone_tbl (tutorial_id, 


ae tutorial_title, 
aE tutorial_author, 
-> submission_date) 
-> SELECT tutorial_id, tutorial_title, 

-> tutorial_author, submission_date, 


-> FROM tutorials_tbl; 
Query OK, 3 rows affected (0.07 sec) 
Records: 3 Duplicates: 0 Warnings: 0 


执行 以 上 步骤 后 ， 你 将 完整 的 复制 表 ， 包 括 表 结构 及 表 数 据 。 


AttrnodeValue' is deprecated. Please use value' instead. adsbygoogle.js:32 


MySQL 元 数据 


你 可 能 想 知道 MySQL 以 下 三 种 信息 : 


e 查询 结果 信息 : SELECT, UPDATE 或 DELETE 语 句 影响 的 记录 数 。 
e 数据 库 和 数据 表 的 信息 : 包含 了 数据 库 及 数据 表 的 结构 信息 。 
e MySQL 服务器 信息 : 包含 了 数据 库 服务 器 的 当前 状态 。 版 本 号 等 。 


在 MySQL 的 命令 提示 符 中 ， 我 们 可 以 很 容易 的 获取 以 上 服务 器 信息 。 但 如 果 使 用 
Perl 或 PHP 等 脚本 语言 ， 你 就 需要 调用 特定 的 接口 函数 来 获取 。 接 下 来 我 们 会 详细 


介绍 。 


获取 查询 语句 影响 的 记录 数 


PERL 实例 
在 DBI 脚本 中 ， 语句 影响 的 记录 数 通 过 画 数 do( ) 或 execute( ) 返 回 : 


# 方法 1 

# 使 用 do( ) 执行 $query 

my $count = $dbh->do ($query); 

# 如 果 发 生 错误 会 输出 0 

printf "%d rows were affected\n", (defined ($count) ? $count : 0); 


# 方法 2 

# 使 用 prepare( ) 及 execute( ) 执行 $query 

my $sth = $dbh->prepare ($query); 

my $count = $sth->execute ( ); 

printf "%d rows were affected\n", (defined ($count) ? $count : 0); 


eee) | 
PHP 实例 
在 PHP 中 ， 你 可 以 使 用 mysql_affected _rows( ) 函数 来 获取 查询 语句 影响 的 记录 
数 。 

$result_id = mysql_query ($query, $conn_id); 

# 如 果 查 询 失 败 返 回 


$count = ($result_id ? mysql_affected_rows ($conn_id) : 0); 
print ("$count rows were affected\n"); 


数据 库 和 数据 表 列 表 


你 可 以 很 容易 的 在 MySQL 服 务 器 中 获取 数据 库 和 数据 表 列 表 。 如 果 你 没有 足够 的 
权限 ， 结 果 将 返回 null, 


你 也 可 以 使 用 SHOW TABLES 或 SHOW DATABASES 语句 来 获取 数据 库 和 数据 
表 列 表 。 
PERL 实例 


# 获取 当前 数据 库 中 所 有 可 用 的 表 。 

my @tables = $dbh->tables ( ); 

foreach $table (@tables ){ 
print "Table Name $table\n"; 


} 


PHP 实例 


<?php 
$con = mysql_connect("localhost", "userid", "password"); 
if (!$con) 


die('Could not connect: ' . mysql_error()); 


} 

$db_list = mysql_list_dbs($con); 

while ($db = mysql_fetch_object($db_list) ) 
echo $db->Database . "<br />"; 


mysql_close($con); 
?> 


获取 服务 器 元 数据 


以 下 命令 语句 可 以 在 MySQL 的 命 信 提 示 符 使 用 ， 也 可 以 在 脚本 中 使 用 ， 如 PHP 肢 
本 。 


命令 
SELECT VERSION( ) 
SELECT DATABASE( ) 
SELECT USER( ) 
SHOW STATUS 


SHOW VARIABLES 


描 ; 


Ef 


服务 器 版 本 信息 

当前 数据 库 名 (或 者 返回 空 ) 
当前 用 户 名 

服务 器 状态 
服务 器 配置 变量 


MySQL 序列 使 用 


MySQL 序 列 是 一 组 整数 : 1, 2, 3, ...， 由 于 一 张 数据 表 只 能 有 一 个 字段 自 增 主 键 ， 
如 果 你 想 实 现 其 他 字段 也 实现 自动 增加 ， 就 可 以 使 用 MySQL 序 列 来 实现 。 


本 章 我 们 将 介绍 如 何 使 用 MySQL 的 序列 。 


使 用 AUTO_INCREMENT 


MySQL 中 最 简单 使 用 序列 的 方法 就 是 使 用 MySQL AUTO_INCREMENT 来 定义 
列 。 


实例 


以 下 实例 中 创建 了 数据 表 insect， insect 中 id 无 需 指定 值 可 实现 自动 增长 。 


mysql> CREATE TABLE insect 
> 
-> id INT UNSIGNED NOT NULL AUTO_INCREMENT, 
-> PRIMARY KEY (id), 
-> name VARCHAR(30) NOT NULL, # type of insect 
-> date DATE NOT NULL, # date collected 
-> Origin VARCHAR(30) NOT NULL # where collected 
); 
Query OK, © rows affected (0.02 sec) 
mysql> INSERT INTO insect (id,name,date,origin) VALUES 
-> (NULL, 'housefly', '2001-09-10', 'kitchen'), 
-> (NULL, 'millipede', '2001-09-10', 'driveway'), 
-> (NULL, 'grasshopper', '2001-09-10', 'front yard'); 
Query OK, 3 rows affected (0.02 sec) 
Records: 3 Duplicates: 0 Warnings: 0 
mysql> SELECT * FROM insect ORDER BY id; 


Ton a Seba eee eS SE en Sees + 
| id | name | date | origin | 
a aoao ne PSs Ss PSS So es + 
| 1 | housefly | 2001-09-10 | kitchen | 
| 2 | millipede | 2001-09-10 | driveway | 
| 3 | grasshopper | 2001-09-10 | front yard | 
eS See Se See Se aaan eon Se + 


3 rows in set (0.00 sec) 


获取 AUTO_INCREMENT 值 


在 MySQL 的 客户 端 中 你 可 以 使 用 SQL 中 的 LAST_INSERT ID( ) 函数 来 获取 最 后 的 
插入 表 中 的 自 增 列 的 值 。 


在 PHP 或 PERL 脚 本 中 也 提供 了 相应 的 函数 来 获取 最 后 的 插入 表 中 的 自 增 列 的 值 。 


PERL 实 例 
使 用 mysql_insertid 属性 来 获取 AUTO_INCREMENT 的 值 。 实例 如 下 : 


$dbh->do ("INSERT INTO insect (name, date, origin) 
VALUES('moth', '2001-09-14', 'windowsill')"); 
my $seq = $dbh->{mysql_insertid}; 


PHP & I 


PHP 通过 mysql_insert_id ORAK # ERAN IT ATE ASQLiZ A 
AUTO_INCREMENT 列 的 值 。 


mysql_query ("INSERT INTO insect (name, date, origin) 
VALUES('moth', '2001-09-14', 'windowsill')", $conn_id); 
$seq = mysgql_insert_id ($conn_id); 


HEFJI 


如 果 你 删除 了 数据 表 中 的 多 条 记录 ， 并 希望 对 剩 下 数据 的 AUTO_ Do 
行 重新 排列 ， 那 么 你 可 以 通过 删除 自 增 的 列 ， 然 后 重新 添加 来 实现 。 不 过 该 操作 要 
非常 小 心 ， 如 果 在 删除 的 同时 又 有 新 记录 添加 ， BB “操作 如 下 
Pa : 


mysql> ALTER TABLE insect DROP id; 

mysql> ALTER TABLE insect 
-> ADD id INT UNSIGNED NOT NULL AUTO_INCREMENT FIRST, 
-> ADD PRIMARY KEY (id); 


Xe he UBF aa 


一 般 情况 下 序列 的 开始 值 为 1， 但 如 果 你 需要 指定 一 个 开始 值 100， 那 我 们 可 以 通过 
以 下 语句 来 实现 : 


mysql> CREATE TABLE insect 
-> ( 
-> id INT UNSIGNED NOT NULL AUTO_INCREMENT = 100, 
-> PRIMARY KEY (id), 
-> name VARCHAR(30) NOT NULL, # type of insect 
-> date DATE NOT NULL, # date collected 
-> Origin VARCHAR(30) NOT NULL # where collected 


); 


或 者 你 也 可 以 在 表 创 建成 功 后 ， 通 过 以 下 语句 来 实现 : 


mysql> ALTER TABLE t AUTO_INCREMENT = 100; 


MySQL 义理 重复 数据 


有 些 MySQL 数据 表 中 可 能 存在 重复 的 记录 ， 有 些 情况 我 们 允许 重复 数据 的 存在 ， 
但 有 时 候 我 们 也 需要 删除 这 些 重复 的 数据 。 


本 章节 我 们 将 为 大 家 介绍 如 何 防止 数据 表 出 现 重复 数据 及 如 何 删 除数 据 表 中 的 重复 


o 


防止 表 中 出 现 重 复数 据 


你 可 以 在 MySQL 数 据 表 中 设置 指定 的 字段 为 PRIMARY KEY (主键 ) 或 者 
UNIQUE (唯一 ) 索引 来 保证 数据 的 唯一 性 。 


让 我 们 尝试 一 个 实例 : 下 表 中 无 索引 及 主键 ， 所 以 该 表 人 允许 出 现 多 条 重复 记录 。 


CREATE TABLE person_tbl 


( 
first_name CHAR(20), 
last_name CHAR(20), 
sex CHAR(10) 


); 


如 果 你 想 设 置 表 中 字段 first name，last_name 数 据 不 能 重复 ， 你 可 以 设置 双 主 键 模 
式 来 设置 数据 的 唯一 性 ， 如 果 你 设置 了 双 主 键 ， 那 么 那个 键 的 默认 值 不 能 为 
NULL， 可 设置 为 NOT NULL。 如 下 所 示 : 


CREATE TABLE person_tbl 


( 
first_name CHAR(20) NOT NULL, 
last_name CHAR(20) NOT NULL, 
sex CHAR(10), 
PRIMARY KEY (last_name, first_name) 


); 


如 果 我 们 设置 了 唯一 索引 ， 那 么 在 插入 重复 数据 时 ，SQL 语 句 将 无 法 执行 成 功 ,并 抛 
出 错 。 


INSERT IGNORE INTOSINSERT INTO 的 区 别 就 是 INSERT IGNORE 会 忽略 数据 
库 中 已 经 存在 的 数据 ， 如 果 数 据 库 没 有 数据 ， 就 插入 新 的 数据 ， 如 果 有 数据 的 话 就 
跳 过 这 条 数据 。 这 样 就 可 以 保留 数据 库 中 已 经 存在 数据 ， 达 到 在 间隙 中 插入 数据 的 
目 的 。 


以 下 实例 使 用 了 INSERT IGNORE INTO， 执 行 后 不 会 出 错 ， 也 不 会 向 数据 表 中 插 
入 重复 数据 : 


mysql> INSERT IGNORE INTO person_tbl (last_name, first_name) 
-> VALUES( 'Jay', 'Thomas'); 

Query OK, 1 row affected (0.00 sec) 

mysql> INSERT IGNORE INTO person_tbl (last_name, first_name) 
-> VALUES( 'Jay', 'Thomas'); 

Query OK, © rows affected (0.00 sec) 


INSERT IGNORE INTO 当 插 和 人 数据 时 ， 在 设置 了 记录 的 唯一 性 后 ， 如 果 插 人 重复 
数据 ， 将 不 返回 错误 ， 只 以 警告 形式 返回 。 而 REPLACE INTO into 如 果 存 在 
primary 或 unique 相 同 的 记录 ， 则 先 删 除 掉 。 再 插入 新 记录 。 


另 一 种 设置 数据 的 唯一 性 方法 是 添加 一 个 UNIQUE 索 引 ， 如 下 所 示 : 


CREATE TABLE person_tbl 


( 
first_name CHAR(20) NOT NULL, 


last_name CHAR(20) NOT NULL, 

sex CHAR(10) 

UNIQUE (last_name, first_name) 
); 


统计 重复 数据 
以 下 我 们 将 统计 表 中 first_name 和 last_name 的 重复 记录 数 : 


mysql> SELECT COUNT(*) as repetitions, last_name, first_name 
-> FROM person_tbl 
-> GROUP BY last_name, first_name 
-> HAVING repetitions > 1; 


以 上 查询 语句 将 返回 person thl 表 中 重复 的 记录 数 。 一 般 情况 下 ， 查 询 重 复 的 
值 ， 请 执行 以 下 操作 : 


确定 哪 一 列 包含 的 值 可 能 会 重复 。 

在 列 选择 列表 使 用 COUNT(*) 列 出 的 那些 列 。 
在 GROUP BY 子 句 中 列 出 的 列 。 
HAVING 子 句 设 置 重 复数 大 于 1。 


过 滤 重 复数 据 


如 果 你 需要 读 取 不 重复 的 数据 可 以 在 SELECT 语句 中 使 用 DISTINCT 关键 字 来 过 
滤 重 复数 据 。 


mysql> SELECT DISTINCT last_name, first_name 
-> FROM person_tbl 
-> ORDER BY last_name; 


你 也 可 以 使 用 GROUP BY 来 读 取 数 据 表 中 不 重复 的 数据 : 


mysql> SELECT last_name, first_name 
-> FROM person_tbl 
-> GROUP BY (last_name, first_name); 


删除 重复 数据 
如 果 你 想 删除 数据 表 中 的 重复 数据 ， 你 可 以 使 用 以 下 的 SQL 语 句 : 


mysql> CREATE TABLE tmp SELECT last_name, first_name, sex 
> FROM person_tbl; 
-> GROUP BY (last_name, first_name); 
mysql> DROP TABLE person_tbl; 
mysql> ALTER TABLE tmp RENAME TO person_tbl; 


当然 你 也 可 以 在 数据 表 中 添加 INDEX (351) 和 PRIMAY KEY (主键 ) 这 种 简单 
的 方法 来 删除 表 中 的 重复 记录 。 方法 如 下 : 


mysql> ALTER IGNORE TABLE person_tbl 
-> ADD PRIMARY KEY (last_name, first_name); 


MySQL 及 SQL 注入 

如 果 您 通过 网 页 获取 用 户 输 入 的 数据 并 将 其 插入 一 个 MySQL 数 据 库 ， 那 么 就 有 可 能 
发 生 SQL 注 入 安全 的 问题 。 

本 章节 将 为 大 家 介绍 如 何 防止 SQL 注入 ， 并 通过 脚本 来 过 滤 SQL 中 注入 的 字符 。 
所 谓 SQL 注 入 ， 就 是 通过 把 SQL 命令 插入 到 Web 表 单 递交 或 输入 域名 或 页 面 请 求 的 
查询 字符 串 ， 最 终 达 到 欺骗 服务 器 执行 恶意 的 SQL 命 今 。 

我 们 永远 不 要 信任 用 户 的 输入 ， 我 们 必须 认定 用 户 输入 的 数据 都 是 不 安全 的 ， 我 们 
都 需要 对 用 户 输入 的 数据 进行 过 滤 处 理 。 

以 下 实例 中 ， 输 入 的 用 户 名 必须 为 字母 、 数 字 及 下 划 线 的 组 合 ， 且 用 户 名 长 度 为 8 
到 20 个 字符 之 间 : 


if (preg_match("/4\w{8, 20}$/", $_GET['username'], $matches) ) 
{ 


$result = mysql query("SELECT * FROM Users 
WHERE username=$matches[0]"); 


else 


echo "username 输入 异常 "， 


让 我 们 看 下 在 没有 过 滤 特 殊 字符 时 ， 出 现 的 SQL 情 ; 


// 设 定 $name 中 插入 了 我 们 不 需要 的 SQL 语 oe) 
$name = "Qadir'; DELETE FROM users; 
mysql_query("SELECT * FROM users WHERE name='{$name}'"); 


以 上 的 注入 语句 中 ， 我 们 没有 对 $name 的 变量 进行 过 滤 ，$name 中 插入 了 我 们 不 
需要 的 SQL 语句 ， 将 删除 users 表 中 的 所 有 数据 。 


在 PHP 中 的 mysql_query() 是 不 允许 执行 多 个 SQL 语句 的 ， 但 是 在 SQLite 和 
PostgreSQL 是 可 以 同时 执行 多 条 SQL 语句 的 ， 所 以 我 们 对 这 些 用 户 的 数据 需要 进 
行 严 格 的 验证 。 


防止 SQL 注入 ， 我 们 需要 注意 以 下 几 个 要 点 : 


e 1. 永 远 不 要 信任 用 户 的 输入 。 对 用 户 的 输入 进行 校 验 ， 可 以 通过 正则 表达 式 ， 
KAR BY KE ; 对 单 引 号 和 双 "-" 进 行 转 换 等 。 

e。 2. 永 远 不 要 使 用 动态 拼装 sql， 可 以 使 用 参数 化 的 sq| 或 者 直接 使 用 存储 过 程 进行 
数据 查询 存 取 。 

。 3. 永 远 不 要 使 用 管理 员 权 限 的 数据 库 连 接 ， 为 每 个 应 用 使 用 单独 的 权限 有 限 的 


数据 库 连 接 。 
eV 4. 不 要 把 机 密 信 息 直 接 存 放 ， 加 密 或 者 hash 掉 密码 和 敏感 的 信息 。 
@ 5. 应 用 的 异 常 信息 应 该 给 出 尽 可 能 少 的 提示 ， 最 好 使 用 自 定义 的 错 i 误 信 AA 息 对 原 
错误 信息 进行 包装 
° se 仿 测 方法 一 般 采 取 辅 助 软件 或 网 站 平台 3 am, 软件 一 般 采 用 sql 注 
入 检测 工具 jsky， 网 站 平台 就 有 亿 思 网 站 安全 平台 检测 工具 。 MDCSOFT 
SCAN 等 。 采 用 MDCSOFT-IPS 可 以 有 效 的 防御 SQL 注入 ，XSS 攻 击 等 。 


防止 SQL 注入 
在 脚本 语言 ， 如 Perl 和 PHP 你 可 以 对 用 户 输入 的 数据 进行 转 义 从 而 来 防止 SQL 注 


o 


PHPBYMySQL4r tet T mysql_real escape_string KRR} L EIREJA A E 
if (get_magic_quotes_gpc()) 


$name = stripslashes($name) ; 


} 
$name = mysql_real_escape_string($name) ; 
mysql_query("SELECT * FROM users WHERE name='{$name}'"); 


Like 语 句 中 的 注入 


like 查 询 时 ， 如 果 用 户 输入 的 值 有 "" 和 "%"”， 则 会 出 现 这 种 情况 : 用 户 本 来 只 是 想 查 
询 "abcaq"， 查 询 结果 中 却 有 "abcd "、"abcde"、"abcdf" 等 等 ; APES 
询 "30%" GE: 百 分 之 三 十 ) 时 也 会 出 现 问 题 。 


在 PHP 上 脚本 中 我 们 可 以 使 用 addcslashes() 函 数 来 处 理 以 上 情况 ， 如 下 实例 : 


$sub = addcslashes(mysql_real_escape_string("%something_"), "%_"); 
// $sub == \%something\_ 
mysql_query("SELECT * FROM messages WHERE subject LIKE '{$sub}%'"), 


a: 
addcslashes() HAE ENF AAO RHL. 
语法 格式 : 


addcslashes(string,characters) 


参数 描述 


string 必需 。 规 定 要 检查 的 字符 串 。 
characters 可 选 。 规 定 受 addcslashes() 影响 的 字符 或 字符 范围 。 


具体 应 用 可 以 查看 : PHP addcslashes() WAX 


MySQL 导出 数据 


MySQL 中 你 可 以 使 用 SELECT...INTO OUTFILE 语 句 来 简单 的 导出 数据 到 文本 文件 


o 


使 用 SELECT ... INTO OUTFILE 语句 导出 数据 
以 下 实例 中 我 们 将 数据 表 tutorials_tbl 数据 导出 到 /tmp/tutorials.txt 文件 中 : 


mysql> SELECT * FROM tutorials_tbl 
-> INTO OUTFILE '/tmp/tutorials.txt'; 


你 可 以 通过 命令 选项 来 设置 数据 输出 的 指定 格式 ， 以 下 实例 为 导出 CSV 格式 : 


mysql> SELECT * FROM passwd INTO OUTFILE '/tmp/tutorials.txt' 
-> FIELDS TERMINATED BY ',' ENCLOSED BY '"' 
-> LINES TERMINATED BY '\r\n'; 


在 下 面 的 例子 中 ， 生 成 一 个 文件 ， 各 值 用 逗号 隐 开 。 这 种 格式 可 以 被 许多 程序 使 


o 


SELECT a,b,a+b INTO OUTFILE '/tmp/result.text' 
FIELDS TERMINATED BY ',' OPTIONALLY ENCLOSED BY '"' 
LINES TERMINATED BY '\n' 

FROM test_table; 


SELECT... INTO OUTFILE 语句 有 以 下 属性 : 


e LOAD DATAINFILE 是 SELECT ... INTO OUTFILE 的 道 操作 ，SELECT 句 法 。 

为 了 将 一 个 数据 库 的 数据 写 入 一 个 文件 ， 使 用 SELECT ... INTO OUTFILE, > 

了 将 文件 读 回 数据 库 ， 使 用 LOAD DATA INFILE。 

SELECT...INTO OUTFILE 'file_name' 形 式 的 SELECT 可 以 把 被 选择 的 行 写 人 一 

o 该 文件 被 创建 到 服务 器 主机 上 ， 因 此 您 必须 拥有 FILE 权 限 ， 才 能 使 

用 j$} 

。 输出 不 能 是 一 个 已 存在 的 文件 。 防 止 文件 数据 被 自 改 。 

。 你 需要 有 一 个 登陆 服务 器 的 账号 来 检索 文件 。 否 则 SELECT... INTO 
OUTFILE 不 会 起 任何 作用 。 

e 在 UNIX 中 ， 该 文件 被 创建 后 是 可 读 的 ， 权 限 由 MySQL 服 务 器 所 拥有 。 这 意味 
着 ， 虽 然 你 就 可 以 读 取 该 文件 ， 但 可 能 无 法 将 其 删除 。 


导出 表 作为 原始 数据 


mysqldump 是 mysql 用 于 转 存 储 数据 库 的 实用 程序 。 它 主要 产生 一 个 SQL 脚本 ， 其 
中 包含 从 头 重新 创建 数据 库 所 必需 的 命 信 CREATE TABLE INSERTS. 


使 用 mysqldump 导 出 数据 需要 使 用 --tab 选项 来 指定 导出 文件 指定 的 目录 ， 该 目标 
必须 是 可 写 的 。 


以 下 实例 将 数据 表 tutorials_tbl 导出 到 /tmp 目录 中 : 


$ mysqldump -u root -p --no-create-info \ 
--tab=/tmp TUTORIALS tutorials _ tbl 
password ****** 


导出 SQL 格 式 的 数据 
导出 SQL 格 式 的 数据 到 指定 文件 ， 如 下 所 示 : 


$ mysqldump -u root -p TUTORIALS tutorials_tbl > dump .txt 
password ****** 


以 上 命令 创建 的 文件 内 容 如 下 : 


-- MySQL dump 8.23 


-- Host: localhost Database: TUTORIALS 


-- Server version 3.23.58 


-- Table structure for table “tutorials tbl 


CREATE TABLE tutorials_tbl ( 
tutorial_id int(11) NOT NULL auto_increment, 
tutorial_title varchar(100) NOT NULL default '', 
tutorial_author varchar(40) NOT NULL default '', 
submission_date date default NULL, 
PRIMARY KEY (tutorial_id), 
UNIQUE KEY AUTHOR_INDEX (tutorial_author) 

) TYPE=MyISAM; 


-- Dumping data for table ‘tutorials _ tbl. 


INSERT INTO tutorials_tbl 

VALUES (1,'Learn PHP','John Poul', '2007-05-24'); 
INSERT INTO tutorials_tbl 

VALUES (2, 'Learn MySQL', 'Abdul S', '2007-05-24'); 
INSERT INTO tutorials_tbl 

VALUES (3,'JAVA Tutorial', 'Sanjay', '2007-05-06'); 


如 果 你 需要 导出 整个 数据 库 的 数据 ， 可 以 使 用 以 下 命令 : 


$ mysqldump -u root -p TUTORIALS > database_dump.txt 
password ****** 


如 果 需 要 各 份 所 有 数据 库 ， 可 以 使 用 以 下 命令 : 


$ mysqldump -u root -p --all-databases > database_dump.txt 
password ****** 


--all-databases 选项 在 MySQL 3.23.12 及 以 后 版 本 加 入 。 
该 方法 可 用 于 实现 数据 库 的 各 份 策略 。 


AY Ga Rs ES BABE AM 


如 果 你 需要 将 数据 拷贝 至 其 他 的 MySQL 服务 器 上 , 你 可 以 在 mysqldump 命令 中 指 
定数 据 库 名 及 数据 表 。 


在 源 主 机 上 执行 以 下 命令 ， 将 数据 各 份 至 dump.txt 文件 中 : 


$ mysqldump -u root -p database_name table_name > dump.txt 
password ***** 


如 果 完 整备 份 数据 库 ， 则 无 需 使 用 特定 的 表 名 称 。 
如 果 你 需要 将 备份 的 数据 库 导 人 到 MySQL 服 务 器 中 ， 可 以 使 用 以 下 命令 ， 使 用 以 下 
命令 你 需要 确认 数据 库 已 经 创建 : 


$ mysql -u root -p database_name < dump.txt 
password ***** 


你 也 可 以 使 用 以 下 命令 将 导出 的 数据 直接 导 人 到 远程 的 服务 器 上 ， 但 请 确保 两 台 服 务 器 是 柑 
$ mysqldump -u root -p database_name \ 
| mysql -h other-host.com database_name 


aj = 5 
以 上 命令 中 使 用 了 管道 来 将 导出 的 数据 导 人 到 指定 的 远程 主机 上 。 





MySQL 导入 数据 


MySQL 中 可 以 使 用 两 种 简单 的 方式 来 导入 MySQL 导 出 的 数据 。 


使 用 LOAD DATA 导入 数据 


MySQL 中 提供 了 LOAD DATAINFILE 语 句 来 插入 数据 。 以 下 实例 中 将 从 当前 目录 
中 读 取 文件 dump.txt ， 将 该 文件 中 的 数据 插入 到 当前 数据 库 的 mytbl 表 中 。 


mysql> LOAD DATA LOCAL INFILE 'dump.txt' INTO TABLE mytbl; 


如 果 指 定 LOCAL 关 键 词 ， 则 表明 从 客户 主机 上 按 路 径 读 取 文 件 。 如 果 没 有 指定 ， 
则 文件 在 服务 器 上 按 路径 读 取 文 件 。 


你 能 明确 地 在 LOAD DATA 语 句 中 指出 列 值 的 分 隔 符 和 行 尾 标 记 ， 但 是 默认 标记 是 
定位 符 和 换行 符 。 


两 个 命令 的 FIELDS 和 LINES 子 句 的 语法 是 一 样 的。 两 个 子 句 都 是 可 选 的 ， 但 是 
如 果 两 个 同时 被 指定 ，FIELDS 子 句 必须 出 现在 LINES 子 句 之 前 。 


如 果 用 户 指定 一 个 FIELDS 子 句 ， 它 的 子 句 (TERMINATED BY, [OPTIONALLY] 
ENCLOSED BY 和 ESCAPED BY) 也 是 可 选 的 ， 不 过 ， 用 户 必 须 至 少 指定 它们 中 
= 


mysql> LOAD DATA LOCAL INFILE 'dump.txt' INTO TABLE mytbl 
-> FIELDS TERMINATED BY ':' 
-> LINES TERMINATED BY '\r\n'; 


LOAD DATA 默认 情况 下 是 按照 数据 文件 中 列 的 顺序 插入 数据 的 ， 如 果 数 据 文件 中 
的 列 与 插入 表 中 的 列 不 一 致 ， 则 需要 指定 列 的 顺序 。 


如 ， 在 数据 文件 中 的 列 顺序 是 a,b,c， 但 在 插入 表 的 列 顺序 为 blca， 则 数据 导 人 话 
法 如 下 : 


mysql> LOAD DATA LOCAL INFILE 'dump.txt' 
-> INTO TABLE mytbl (b, c, a); 


使 用 mysqlimport 导入 数据 


mysqlimport 客 户 端 提 供 了 LOAD DATAINFILEQL 语 句 的 一 个 命令 行 接口 。 
mysqlimport 的 大 多 数 选项 直接 对 应 LOAD DATAINFILE 子 句 。 


从 文件 dump.txt 中 将 数据 导入 到 mytbl 数据 表 中 , 可 以 使 用 以 下 命令 : 


$ mysqlimport -u root 


-p --local database_name dump.txt 


password ***** 


mysqlimport 命 令 可 以 指定 选项 来 设置 指定 格式 ,命令 语句 格式 如 下 : 


$ mysqlimport -u root -p --local 
--lines-terminated-by="\r\n" 


--fields-terminated-by=":" \ 
database_name dump.txt 


password ***** 


mysqlimport 语句 中 使 用 --columns 选项 来 设置 列 的 顺序 : 


$ mysqlimport -u root -p --local 


--columns=b,c,a \ 


database_name dump.txt 
password ***** 


mysqlimport 的 常用 选项 介绍 


选项 


-d or -- 
delete 


-f or --force 


-i or -- 
ignore 


-| or -lock- 
tables 


-ror - 
replace 


--fields- 
enclosed- 
by= char 


--fields- 
terminated- 
by=char 


--lines- 
terminated- 
by=str 


新 数据 导入 数据 表 中 之 前 删除 数据 数据 表 中 的 所 有 信息 


不 管 是 否 遇 到 错误 ，mysqlimport 将 强制 继续 插入 数据 


mysqlimport 跳 过 或 者 忽略 那些 有 相同 唯一 关键 字 的 行 ， 
件 中 的 数据 将 被 忽略 。 

数据 被 插入 之 前 锁 住 表 ， 这 样 就 防止 了 ， 
户 的 查询 和 更 新 受到 影响 。 


这 个 选项 与 一 选项 的 作用 相反 ; 此 选项 将 替代 表 中 有 相同 唯一 
关键 字 的 记录 。 


导入 文 


你 在 更 新 数据 库 时 ， 用 


指定 文本 文件 中 数据 的 记录 时 以 什么 括 起 的 ， 很 多 情况 下 数据 
以 双 引 号 括 起 。 默认 的 情况 下 数据 是 没有 被 字符 括 起 的 。 


指定 各 个 数据 的 值 之 间 的 分 隔 符 ， 在 句号 分 隔 的 文件 中 ， abet 
是 句号 。 您 可 以 用 此 选项 指定 数据 之 间 的 分 隔 符 。 默认 的 分 隔 符 
是 跳 格 符 (Tab) 


此 选项 指定 文本 文件 中 行 与 行 之 间 数 据 的 分 隔 字符 串 或 者 字符 。 
默认 的 情况 下 mysqlimport 以 newline 为 行 分 隔 符 。 您 可 以 选择 用 
一 个 字符 串 来 替代 一 个 单个 的 字符 : 一 个 新 行 或 者 一 个 回 车 。 


mysdlimport 命 令 常 用 的 选项 还 有 -V 显示 版 本 (version) , -p 提示 输入 密码 
(password) 等 。 


免责 声明 


W3School 提 供 的 内 容 仅 用 于 培训 。 我 们 不 保证 内 容 的 正确 性 。 通 过 使 用 本 站 内 容 
随 之 而 来 的 风险 与 本 站 无 关 。W3School 简体 中 文 版 的 所 有 内 容 仅 供 测试 ， 对 任何 
法 律 问题 及 风险 不 承担 任何 责任 。 


